Issue Details (XML | Word | Printable)

Key: DERBY-3813
Type: Bug Bug
Status: Resolved Resolved
Resolution: Fixed
Priority: Major Major
Assignee: Kathey Marsden
Reporter: Stan Bradbury
Votes: 0
Watchers: 1
Operations

If you were logged in you would be able to see more operations.
Derby

Derby tests for the existance of BigDecimal methods toPlainString and bdPrecison but does not check if they were found before using them.

Created: 02/Aug/08 12:46 AM   Updated: 04/May/09 06:22 PM
Return to search
Component/s: SQL
Affects Version/s: 10.3.3.0
Fix Version/s: 10.3.3.1, 10.4.2.0, 10.5.1.1

Time Tracking:
Not Specified

File Attachments:
  Size
Text File Licensed for inclusion in ASF works DERBY-3813_diff.txt 2008-08-05 04:10 PM Kathey Marsden 0.8 kB
Environment: Using a VM that is a subset of Java 5 without full BigDecimal support

Resolution Date: 08/Aug/08 03:47 AM


 Description  « Hide
I received the following problem report:

Having failures running with our VM that is a subset of Java 5. Failure is with the handling of the BigDecimal class in the org.apache.derby.iapi.types.SQLDecimal class.

The SQLDecimal class does reflection to determine if the methods toPlainString and bdPrecison are present or not
 private static final Method toPlainString;
    private static final Method bdPrecision;
    static {
        Method m;
        try {
            m = BigDecimal.class.getMethod("toPlainString", null);
        } catch (NoSuchMethodException e) {
            m = null;
        }
        toPlainString = m;
        try {
            m = BigDecimal.class.getMethod("precision", null);
        } catch (NoSuchMethodException e) {
            m = null;
        }
        bdPrecision = m;
    }

however, when it comes time to use them, it does not check whether it found the methods, but rather the JVM level, equating JVM specification level with method existence

public String getString()
{
BigDecimal localValue = getBigDecimal();
if (localValue == null)
return null;
else if (JVMInfo.JDK_ID < JVMInfo.J2SE_15)
return localValue.toString();
        else
        {
            // use reflection so we can still compile using JDK1.4
            // if we are prepared to require 1.5 to compile then this can be a direct call
            try {
                return (String) toPlainString.invoke(localValue, null);
            } catch (IllegalAccessException e) {


and
private static int getWholeDigits(BigDecimal decimalValue)
{
        /**
         * if ONE > abs(value) then the number of whole digits is 0
         */
        decimalValue = decimalValue.abs();
        if (ONE.compareTo(decimalValue) == 1)
        {
            return 0;
        }
        
        if (JVMInfo.JDK_ID >= JVMInfo.J2SE_15)
{
// use reflection so we can still compile using JDK1.4
// if we are prepared to require 1.5 to compile then this can be a
// direct call
try {
// precision is the number of digits in the unscaled value,
// subtracting the scale (positive or negative) will give the
// number of whole digits.
int precision = ((Integer) bdPrecision.invoke(decimalValue,
null)).intValue();
return precision - decimalValue.scale();

Since the JVM is claiming 1.5 specification, the code assumes the methods exist, and result in NullPointerExceptions at the low level Derby class, resulting in SQLExceptions thrown to the caller.

>>>> ***Created table: dish
***Created table: VNMEDICALRECORD
Fail to make the query to database:
java.lang.NullPointerException
        at org.apache.derby.iapi.types.SQLDecimal.getString(Unknown Source)
        at org.apache.derby.iapi.types.DataType.getTraceString(Unknown Source)
        at org.apache.derby.impl.sql.GenericParameter.toString(Unknown Source)
        at org.apache.derby.impl.sql.GenericParameterValueSet.toString(Unknown Source)
        at org.apache.derby.impl.sql.conn.GenericStatementContext.appendErrorInfo(Unknown Source)
        at org.apache.derby.iapi.services.context.ContextManager.cleanupOnError(Unknown Source)
        at org.apache.derby.impl.jdbc.TransactionResourceImpl.handleException(Unknown Source)
        at org.apache.derby.impl.jdbc.EmbedConnection.handleException(Unknown Source)
        at org.apache.derby.impl.jdbc.ConnectionChild.handleException(Unknown Source)
        at org.apache.derby.impl.jdbc.EmbedStatement.executeStatement(Unknown Source)
        at org.apache.derby.impl.jdbc.EmbedPreparedStatement.executeStatement(Unknown Source)
        at org.apache.derby.impl.jdbc.EmbedPreparedStatement.execute(Unknown Source)
        at com.ibm.rcp.samples.derby.Derby.updateDB(Derby.java:175)
        at com.ibm.rcp.samples.derby.Derby.run(Derby.java:89)
        at java.lang.Thread.run(Unknown Source)
Fail to close:
java.sql.SQLException: Cannot issue rollback in a nested connection when there is a pending operation in the parent connection.
        at org.apache.derby.impl.jdbc.SQLExceptionFactory.getSQLException(Unknown Source)
        at org.apache.derby.impl.jdbc.Util.generateCsSQLException(Unknown Source)
        at org.apache.derby.impl.jdbc.TransactionResourceImpl.wrapInSQLException(Unknown Source)
        at org.apache.derby.impl.jdbc.TransactionResourceImpl.handleException(Unknown Source)
        at org.apache.derby.impl.jdbc.EmbedConnection.handleException(Unknown Source)
        at org.apache.derby.impl.jdbc.EmbedConnection.close(Unknown Source)
        at org.apache.derby.impl.jdbc.EmbedConnection.close(Unknown Source)
        at com.ibm.rcp.samples.derby.Derby.updateDB(Derby.java:258)
        at com.ibm.rcp.samples.derby.Derby.run(Derby.java:89)
        at java.lang.Thread.run(Unknown Source)
Caused by: ERROR X0Y67: Cannot issue rollback in a nested connection when there is a pending operation in the parent connection.
        at org.apache.derby.iapi.error.StandardException.newException(Unknown Source)
        at org.apache.derby.impl.sql.conn.GenericLanguageConnectionContext.doRollback(Unknown Source)
        at org.apache.derby.impl.sql.conn.GenericLanguageConnectionContext.userRollback(Unknown Source)
        at org.apache.derby.impl.jdbc.TransactionResourceImpl.rollback(Unknown Source)
        ... 5 more


Making the following change would fix the problem of running on our VM
Instead of this test
        if (JVMInfo.JDK_ID >= JVMInfo.J2SE_15)
test if (toPlainString == null), or if (bdPrecision == null) or equivalent, this should achieved the same results


 All   Comments   Work Log   Change History   Subversion Commits      Sort Order: Ascending order - Click to sort in descending order
Kathey Marsden added a comment - 04/Aug/08 09:15 PM

Kathey Marsden added a comment - 05/Aug/08 04:10 PM
Here is a patch to correct this issue. I reproduced the NPE with the JVM in question and verified that this fixes it. I ran suites.All and derbyall on weme 6.1, suites.All on jdk142. Still need to run more tests, but here is the patch for review.


Kathey Marsden added a comment - 06/Aug/08 04:50 PM
I plan to go ahead and check this in and backport to 10.4 and 10.3. I ran suites.All and derbyall with weme 6.1, jdk142, and jdk 1.6. I ran the repro with the jvm in question and saw the NullPointerException and saw it run clean with the fix, so this is likely to fix the problem for this particular case. That doesn't say anything for how Derby will run in general against a 1.5 subset.


Berthold Lebert added a comment - 07/Aug/08 11:32 PM
What are the reasons you go through reflection to use toPlainString() and precision() if in the Java 1.4 case toString() and your code to determine the precision seem to be just fine?

Kathey Marsden added a comment - 08/Aug/08 03:47 AM
Committed fix to trunk, 10.4 and 10.5 codeline. Dan do you know the answer to Berthold's question?


Daniel John Debrunner added a comment - 08/Aug/08 03:57 PM
The issue is that the format of the value issued by toString() changed from Java 1.4 to 1.5, an unusual incompatibility introduced into java.

Thus the code to determine precision from toString() works in 1.4 but will not work in 1.5, would have to use toPlainString() instead, but since 1.5 introduced a precision() method it makes most sense to use that in 1.5.