OPENJPA-1794: Return null, rather than 0, on MAX function - back ported to 2.1.x Jeremy Bauer's commit to trunk.

git-svn-id: https://svn.apache.org/repos/asf/openjpa/branches/2.1.x@1529241 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Heath Thomann 2013-10-04 17:58:15 +00:00
parent edfa79ca39
commit e8f8874a72
12 changed files with 600 additions and 8 deletions

View File

@ -23,8 +23,9 @@ package org.apache.openjpa.jdbc.kernel.exps;
* *
* @author Abe White * @author Abe White
*/ */
@SuppressWarnings("serial")
class Avg class Avg
extends UnaryOp { extends NullableAggregateUnaryOp { // OPENJPA-1794
/** /**
* Constructor. Provide the value to operate on. * Constructor. Provide the value to operate on.
@ -41,4 +42,3 @@ class Avg
return true; return true;
} }
} }

View File

@ -23,8 +23,9 @@ package org.apache.openjpa.jdbc.kernel.exps;
* *
* @author Abe White * @author Abe White
*/ */
@SuppressWarnings("serial")
class Max class Max
extends UnaryOp { extends NullableAggregateUnaryOp { // OPENJPA-1794
/** /**
* Constructor. Provide the value to operate on. * Constructor. Provide the value to operate on.

View File

@ -23,8 +23,9 @@ package org.apache.openjpa.jdbc.kernel.exps;
* *
* @author Abe White * @author Abe White
*/ */
@SuppressWarnings("serial")
class Min class Min
extends UnaryOp { extends NullableAggregateUnaryOp { // OPENJPA-1794
/** /**
* Constructor. Provide the value to operate on. * Constructor. Provide the value to operate on.

View File

@ -0,0 +1,46 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.openjpa.jdbc.kernel.exps;
/**
* OPENJPA-1794
* An aggregate unary operation that can indicate whether a null value from the data store
* should be returned as null.
*/
@SuppressWarnings("serial")
public abstract class NullableAggregateUnaryOp extends UnaryOp {
public NullableAggregateUnaryOp(Val val) {
super(val);
}
public NullableAggregateUnaryOp(Val val, boolean noParen) {
super(val, noParen);
}
@Override
protected boolean nullableValue(ExpContext ctx, ExpState state) {
// If this is a simple operator (no joins involved), check compatibility to determine
// whether 'null' should be returned for the aggregate operation
if (ctx != null && ctx.store != null && (state.joins == null || state.joins.isEmpty())) {
return ctx.store.getConfiguration().getCompatibilityInstance().getReturnNullOnEmptyAggregateResult();
}
return false;
}
}

View File

@ -25,8 +25,9 @@ import org.apache.openjpa.kernel.Filters;
* *
* @author Abe White * @author Abe White
*/ */
@SuppressWarnings("serial")
class Sum class Sum
extends UnaryOp { extends NullableAggregateUnaryOp { // OPENJPA-1794
/** /**
* Constructor. Provide the value to operate on. * Constructor. Provide the value to operate on.

View File

@ -119,8 +119,13 @@ abstract class UnaryOp
throws SQLException { throws SQLException {
Object value = res.getObject(this, JavaSQLTypes.JDBC_DEFAULT, null); Object value = res.getObject(this, JavaSQLTypes.JDBC_DEFAULT, null);
Class<?> type = getType(); Class<?> type = getType();
if (value == null && (type.isPrimitive() || Number.class.isAssignableFrom(type))) { if (value == null) {
value = Filters.getDefaultForNull(Filters.wrap(type)); if (nullableValue(ctx, state)) { // OPENJPA-1794
return null;
}
else if (type.isPrimitive() || Number.class.isAssignableFrom(type)) {
value = Filters.getDefaultForNull(Filters.wrap(type));
}
} }
return Filters.convert(value, type); return Filters.convert(value, type);
} }
@ -171,5 +176,10 @@ abstract class UnaryOp
_val.acceptVisit(visitor); _val.acceptVisit(visitor);
visitor.exit(this); visitor.exit(this);
} }
}
// OPENJPA-1794
protected boolean nullableValue(ExpContext ctx, ExpState state) {
return false;
}
}

View File

@ -73,6 +73,7 @@ public class Compatibility {
private boolean _overrideContextClassloader = false; //OPENJPA-1993 private boolean _overrideContextClassloader = false; //OPENJPA-1993
private boolean _filterPCRegistryClasses = false; // OPENJPA-2288 private boolean _filterPCRegistryClasses = false; // OPENJPA-2288
private boolean _useListAttributeForArrays = true; private boolean _useListAttributeForArrays = true;
private boolean _returnNullOnEmptyAggregateResult = false; // OPENJPA-1794
/** /**
* Whether to require exact identity value types when creating object * Whether to require exact identity value types when creating object
@ -681,5 +682,39 @@ public class Compatibility {
public void setUseListAttributeForArrays(boolean useListAttribute ) { public void setUseListAttributeForArrays(boolean useListAttribute ) {
_useListAttributeForArrays = useListAttribute; _useListAttributeForArrays = useListAttribute;
} }
/**
* This property is used to specify whether the aggregate query functions
* SUM, AVG, MAX, and MIN return null if there is no query result. This will occur
* if no rows are returned for the specified query predicate. The default is
* false, meaning that 0 will be returned for functions operating on numeric
* data.
*
* In compliance with the JPA specification, the default value is true.
*
* @return true if the result of an aggregate with an empty query result returns null.
* @since
*
*/
public boolean getReturnNullOnEmptyAggregateResult() {
return _returnNullOnEmptyAggregateResult;
}
/**
* This property is used to specify whether the aggregate query functions
* SUM, AVG, MAX, and MIN return null if there is no query result. This will occur
* if no rows are returned for the specified query predicate. The default is
* false, meaning that 0 will be returned for functions operating on numeric
* data.
*
* In compliance with the JPA specification, the default value is true.
*
* @since
* @param returnNullOnAggregate whether OpenJPA will return null for aggregate
* expressions when the query result is empty.
*/
public void setReturnNullOnAggregateResult(boolean returnNullOnEmptyAggregateResult) {
_returnNullOnEmptyAggregateResult = returnNullOnEmptyAggregateResult;
}
} }

View File

@ -0,0 +1,160 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.openjpa.jira1794;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.Table;
@Entity
@Table(name = "j1794_ae")
public class AggEntity {
@Id
@GeneratedValue
private int id;
private short pshortVal;
private Short shortVal;
private int pintVal;
private Integer intVal;
private long plongVal;
private Long longVal;
private float pfloatVal;
private Float floatVal;
private double pdblVal;
private Double dblVal;
private String stringVal;
public void setId(int id) {
this.id = id;
}
public int getId() {
return id;
}
public void setPshortVal(short pshortVal) {
this.pshortVal = pshortVal;
}
public short getPshortVal() {
return pshortVal;
}
public void setShortVal(Short pShortVal) {
this.shortVal = pShortVal;
}
public Short getShortVal() {
return shortVal;
}
public void setPintVal(int pintVal) {
this.pintVal = pintVal;
}
public int getPintVal() {
return pintVal;
}
public void setIntVal(Integer intVal) {
this.intVal = intVal;
}
public Integer getIntVal() {
return intVal;
}
public void setPlongVal(long plongVal) {
this.plongVal = plongVal;
}
public long getPlongVal() {
return plongVal;
}
public void setLongVal(Long longVal) {
this.longVal = longVal;
}
public Long getLongVal() {
return longVal;
}
public void setPfloatVal(float pfloatVal) {
this.pfloatVal = pfloatVal;
}
public float getPfloatVal() {
return pfloatVal;
}
public void setFloatVal(Float floatVal) {
this.floatVal = floatVal;
}
public Float getFloatVal() {
return floatVal;
}
public void setPdblVal(double pdblVal) {
this.pdblVal = pdblVal;
}
public double getPdblVal() {
return pdblVal;
}
public void setDblVal(Double dblVal) {
this.dblVal = dblVal;
}
public Double getDblVal() {
return dblVal;
}
public void setStringVal(String stringVal) {
this.stringVal = stringVal;
}
public String getStringVal() {
return stringVal;
}
public void init() {
setPshortVal((short) 1);
setShortVal(Short.valueOf((short) 1));
setIntVal(1);
setPintVal(1);
setLongVal(1L);
setPlongVal(1L);
setDblVal(1d);
setPdblVal(1d);
setFloatVal(1f);
setPfloatVal(1f);
setStringVal("1");
}
}

View File

@ -0,0 +1,41 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
/**
* Generated by OpenJPA MetaModel Generator Tool.
**/
package org.apache.openjpa.jira1794;
import javax.persistence.metamodel.SingularAttribute;
@javax.persistence.metamodel.StaticMetamodel
(value=org.apache.openjpa.jira1794.AggEntity.class)
public class AggEntity_ {
public static volatile SingularAttribute<AggEntity,Short> pshortVal;
public static volatile SingularAttribute<AggEntity,Short> shortVal;
public static volatile SingularAttribute<AggEntity,Integer> pintVal;
public static volatile SingularAttribute<AggEntity,Integer> intVal;
public static volatile SingularAttribute<AggEntity,Long> plongVal;
public static volatile SingularAttribute<AggEntity,Long> longVal;
public static volatile SingularAttribute<AggEntity,Float> pfloatVal;
public static volatile SingularAttribute<AggEntity,Float> floatVal;
public static volatile SingularAttribute<AggEntity,Double> pdblVal;
public static volatile SingularAttribute<AggEntity,Double> dblVal;
public static volatile SingularAttribute<AggEntity,String> stringVal;
}

View File

@ -0,0 +1,237 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.openjpa.jira1794;
import java.util.List;
import javax.persistence.EntityManager;
import javax.persistence.Query;
import javax.persistence.criteria.CriteriaBuilder;
import javax.persistence.criteria.CriteriaQuery;
import javax.persistence.criteria.Expression;
import javax.persistence.criteria.Path;
import javax.persistence.criteria.Root;
import javax.persistence.metamodel.Metamodel;
import javax.persistence.metamodel.SingularAttribute;
import org.apache.openjpa.persistence.test.SingleEMFTestCase;
/**
* OPENJPA-1794 Verifies the return value of aggregate functions when a query
* result set is empty. In this set of variations, the compatibility flag is not
* set so null is expected.
*/
public class TestAggregateFunctions extends SingleEMFTestCase {
private static final int MAX = 0;
private static final int MIN = 1;
private static final int SUM = 2;
private static final String[] numericAggregateFunctions = { "MAX", "AVG",
"MIN", "SUM" };
private static final String[] stringAggregateFunctions = { "MAX", "MIN" };
private static final String[] numericAttributes = { "ae.pintVal",
"ae.intVal", "ae.shortVal", "ae.pshortVal", "ae.pintVal",
"ae.intVal", "ae.plongVal", "ae.longVal", "ae.pfloatVal",
"ae.floatVal", "ae.pdblVal", "ae.dblVal" };
@Override
public void setUp() {
super.setUp(CLEAR_TABLES,
"openjpa.Compatibility", "ReturnNullOnAggregateResult=true", //OPENJPA-1794
AggEntity.class);
}
protected boolean nullResultExpected() {
return true;
}
public void testAggregateJPQL() {
EntityManager em = emf.createEntityManager();
// Verify all numeric types for all aggregate functions return null
// if there is no query result
verifyResult(em, numericAggregateFunctions, numericAttributes, true);
// Verify a string for all applicable aggregate functions return null
// if there is no query result
verifyResult(em, stringAggregateFunctions,
new String[] { "ae.stringVal" }, true, true);
// Add a row to the table and re-test
AggEntity ae = new AggEntity();
ae.init();
em.getTransaction().begin();
em.persist(ae);
em.getTransaction().commit();
// Verify all numeric types for all aggregate functions return a
// non-null value when there is a query result
verifyResult(em, numericAggregateFunctions, numericAttributes, false);
// Verify string types for all applicable aggregate functions return a
// non-null value when there is a query result
verifyResult(em, stringAggregateFunctions,
new String[] { "ae.stringVal" }, false);
em.close();
}
public void testAggregateCriteria() {
EntityManager em = emf.createEntityManager();
Metamodel mm = emf.getMetamodel();
mm.getEntities();
Query q = null;
// Verify all types of criteria query that return a Numeric type
for (int agg = MAX; agg <= SUM; agg++) {
CriteriaQuery<Short> cqs = buildNumericCriteriaQuery(em,
Short.class, AggEntity_.shortVal, agg);
q = em.createQuery(cqs);
verifyQueryResult(q, true);
cqs = buildNumericCriteriaQuery(em, Short.class,
AggEntity_.pshortVal, agg);
q = em.createQuery(cqs);
verifyQueryResult(q, true);
CriteriaQuery<Integer> cqi = buildNumericCriteriaQuery(em,
Integer.class, AggEntity_.intVal, agg);
q = em.createQuery(cqi);
verifyQueryResult(q, true);
cqi = buildNumericCriteriaQuery(em, Integer.class,
AggEntity_.pintVal, agg);
q = em.createQuery(cqi);
verifyQueryResult(q, true);
CriteriaQuery<Float> cqf = buildNumericCriteriaQuery(em,
Float.class, AggEntity_.floatVal, agg);
q = em.createQuery(cqf);
verifyQueryResult(q, true);
cqf = buildNumericCriteriaQuery(em, Float.class,
AggEntity_.pfloatVal, agg);
q = em.createQuery(cqi);
verifyQueryResult(q, true);
CriteriaQuery<Double> cqd = buildNumericCriteriaQuery(em,
Double.class, AggEntity_.dblVal, agg);
q = em.createQuery(cqd);
verifyQueryResult(q, true);
cqd = buildNumericCriteriaQuery(em, Double.class,
AggEntity_.pdblVal, agg);
q = em.createQuery(cqi);
verifyQueryResult(q, true);
}
// Verify AVG criteria query - it strictly returns type 'Double' so
// unlike other aggregates,
// it cannot be handled generically (as Numeric).
CriteriaQuery<Double> cqd = buildAvgCriteriaQuery(em, Double.class,
AggEntity_.dblVal);
q = em.createQuery(cqd);
verifyQueryResult(q, true);
cqd = buildAvgCriteriaQuery(em, Double.class, AggEntity_.pdblVal);
q = em.createQuery(cqd);
verifyQueryResult(q, true);
em.close();
}
private <T extends Number> CriteriaQuery<T> buildNumericCriteriaQuery(
EntityManager em, Class<T> type,
SingularAttribute<AggEntity, T> sa, int at) {
CriteriaBuilder cb = em.getCriteriaBuilder();
CriteriaQuery<T> cq = cb.createQuery(type);
Root<AggEntity> aer = cq.from(AggEntity.class);
Path<T> path = aer.get(sa);
Expression<T> exp = null;
switch (at) {
case MAX:
exp = cb.max(path);
break;
case MIN:
exp = cb.min(path);
break;
case SUM:
exp = cb.sum(path);
break;
}
cq.select(exp);
return cq;
}
private CriteriaQuery<Double> buildAvgCriteriaQuery(EntityManager em,
Class<Double> type, SingularAttribute<AggEntity, Double> sa) {
CriteriaBuilder cb = em.getCriteriaBuilder();
CriteriaQuery<Double> cq = cb.createQuery(type);
Root<AggEntity> aer = cq.from(AggEntity.class);
return cq.select(cb.avg(aer.get(sa)));
}
private void verifyResult(EntityManager em, String[] aggregates,
String[] attributes, boolean expectNull) {
verifyResult(em, aggregates, attributes, expectNull, false);
}
private void verifyResult(EntityManager em, String[] aggregates,
String[] attributes, boolean expectNull, boolean isString) {
for (String func : aggregates) {
for (String attr : attributes) {
// JPQL with aggregate and aggregate in subselect
String sql = "SELECT " + func + "(" + attr + ")"
+ " FROM AggEntity ae WHERE " + attr + " <= "
+ "(SELECT " + func + "("
+ attr.replaceFirst("^ae.", "ae2.")
+ ") FROM AggEntity ae2)";
;
Query q = em.createQuery(sql);
verifyQueryResult(q, expectNull, isString);
}
}
}
private void verifyQueryResult(Query q, boolean emptyRs) {
verifyQueryResult(q, emptyRs, false);
}
private void verifyQueryResult(Query q, boolean emptyRs, boolean isString) {
Object result = q.getSingleResult();
if (!emptyRs && !isString) {
assertNotNull(result);
} else if (isString || nullResultExpected()) {
assertNull(result);
} else {
assertNotNull(result);
}
List<?> resultList = q.getResultList();
assertEquals(1, resultList.size());
if (!emptyRs && !isString) {
assertNotNull(result);
} else if (isString || nullResultExpected()) {
assertNull(result);
} else {
assertNotNull(result);
}
}
}

View File

@ -0,0 +1,41 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.openjpa.jira1794;
/**
* OPENJPA-1794
* Verifies the return value of aggregate functions when a query result
* set is empty. In this variation, the compatibility flag
* is set so 0 is expected.
*/
public class TestCompatAggregateFunctions extends TestAggregateFunctions {
@Override
public void setUp() {
super.setUp(CLEAR_TABLES,
"openjpa.Compatibility", "ReturnNullOnAggregateResult=false",
AggEntity.class);
}
@Override
// In compatibility mode a null result is not expected.
protected boolean nullResultExpected() {
return false;
}
}

View File

@ -379,6 +379,25 @@
</itemizedlist> </itemizedlist>
</para> </para>
</section> </section>
<section id="ReturnNullOnEmptyAggregateResult">
<title>
Return value of aggregate functions in SELECT clause
</title>
<!-- See OPENJPA-1794 for details. -->
<para>
The JPA specification states "If SUM, AVG, MAX, or MIN is used, and there are no values to which the aggregate function can be
applied, the result of the aggregate function is NULL." Prior to this update, OpenJPA incorrectly returned 0 for SUM, AVG, MIN,
and MAX when a state field being aggregated is numeric. This behavior affects both JPQL and Criteria queries. With this update,
OpenJPA will return a null result value for these aggregate functions when a query returns no result.
</para>
<para>
To enable the new behavior of this fix, you need to set the following persistence property in your persistence.xml or when
creating an EntityManagerFactory.
<programlisting>
&lt;property name="openjpa.Compatibility" value="ReturnNullOnAggregateResult=true"/&gt;
</programlisting>
</para>
</section>
</section> </section>
</section> </section>
<section id="jpa_2.1"> <section id="jpa_2.1">