Lucee 6 upgrade issue - ORM cannot cast

We are working on upgrading to Lucee 6 (6.0.3.1) and have come across a few compatibility issues. Please let me know if it would be useful to create a JIRA ticket for these issues. We have not found a workaround for this problem.

The test case below demonstrates one of the class cast problems we’ve seen using the ORM capabilities of Lucee. Additionally, we’ve seen similar class casting problems when hibernate attempts to flush the session at the end of the request. This example works with Lucee 5 but fails on version 6.

I’ve tried the lucee hibernate extension and the latest version of the Ortus ORM extension with multiple versions of Lucee 6. The latest tests have been with:

Lucee 6.2.0.1-SNAPSHOT
Ortus ORM Extension 7.0.0.68-snapshot

Error:

Message class java.lang.String cannot be cast to class java.math.BigDecimal (java.lang.String and java.math.BigDecimal are in module java.base of loader ‘bootstrap’)

Stacktrace
The Error Occurred in C:\work\webroot\lucee6\orm\ormtest2.cfm: line 12
10: minBalance = 10000.023423430;
11: queryStr = “FROM Person WHERE balance > :minBalance”;
12: results = ORMExecuteQuery(queryStr, {minBalance: minBalance});
13:
14: for (person in results) {

Example code:

Application.cfc

component {
    this.name = "LuceeORMExample";

    this.ormEnabled = true;
    this.ormSettings = {
        dbCreate = "update",
        dialect = "org.hibernate.dialect.H2Dialect",
        logSQL = true
    };
    
    this.datasources["h2datasource"] = {
        class: "org.h2.Driver", 
        bundleName: "org.lucee.h2", 
        bundleVersion: "2.1.214.0001L",
        connectionString: "jdbc:h2:\temp\h2OrmTest",
        username: "sa",
        password: ""
    };

    this.datasource = "h2datasource";

}

Person.cfc

component persistent="true" table="person" {
    property name="id" fieldtype="id" generator="native";
    property name="firstName" ormtype="string";
    property name="lastName" ormtype="string";
    property name="balance" ormtype="big_decimal";
}

test.cfm

<cfscript>
    // Create a new person
    person = new Person();
    person.setFirstName("John");
    person.setLastName("Doe");
    person.setBalance(12345.123456);
    entitySave(person);
    
    // Query
    minBalance = 10000.023423430;
    queryStr = "FROM Person WHERE balance > :minBalance";
    results = ORMExecuteQuery(queryStr, {minBalance: minBalance});
   
    for (person in results) {
        writeDump(person);
    }
</cfscript>

Here is a stacktrace from the sample code (ormTest2.cfm) running Lucee 6.2.0.60-SNAPSHOT and Ortus ORM 6.5.3.104-snapshot

lucee.runtime.exp.NativeException: class java.lang.String cannot be cast to class java.math.BigDecimal (java.lang.String and java.math.BigDecimal are in module java.base of loader 'bootstrap')
  at org.hibernate.type.descriptor.java.BigDecimalTypeDescriptor.unwrap(BigDecimalTypeDescriptor.java:19)
  at org.hibernate.type.descriptor.sql.DecimalTypeDescriptor$1.doBind(DecimalTypeDescriptor.java:47)
  at org.hibernate.type.descriptor.sql.BasicBinder.bind(BasicBinder.java:73)
  at org.hibernate.type.AbstractStandardBasicType.nullSafeSet(AbstractStandardBasicType.java:276)
  at org.hibernate.type.AbstractStandardBasicType.nullSafeSet(AbstractStandardBasicType.java:271)
  at org.hibernate.param.NamedParameterSpecification.bind(NamedParameterSpecification.java:53)
  at org.hibernate.loader.hql.QueryLoader.bindParameterValues(QueryLoader.java:682)
  at org.hibernate.loader.Loader.bindPreparedStatement(Loader.java:2150)
  at org.hibernate.loader.Loader.prepareQueryStatement(Loader.java:2127)
  at org.hibernate.loader.Loader.executeQueryStatement(Loader.java:2059)
  at org.hibernate.loader.Loader.executeQueryStatement(Loader.java:2037)
  at org.hibernate.loader.Loader.doQuery(Loader.java:956)
  at org.hibernate.loader.Loader.doQueryAndInitializeNonLazyCollections(Loader.java:357)
  at org.hibernate.loader.Loader.doList(Loader.java:2868)
  at org.hibernate.loader.Loader.doList(Loader.java:2850)
  at org.hibernate.loader.Loader.listIgnoreQueryCache(Loader.java:2682)
  at org.hibernate.loader.Loader.list(Loader.java:2677)
  at org.hibernate.loader.hql.QueryLoader.list(QueryLoader.java:540)
  at org.hibernate.hql.internal.ast.QueryTranslatorImpl.list(QueryTranslatorImpl.java:400)
  at org.hibernate.engine.query.spi.HQLQueryPlan.performList(HQLQueryPlan.java:218)
  at org.hibernate.internal.SessionImpl.list(SessionImpl.java:1459)
  at org.hibernate.query.internal.AbstractProducedQuery.doList(AbstractProducedQuery.java:1649)
  at org.hibernate.query.internal.AbstractProducedQuery.list(AbstractProducedQuery.java:1617)
  at ortus.extension.orm.HibernateORMSession.doQueryExecute(HibernateORMSession.java:826)
  at ortus.extension.orm.HibernateORMSession.wrapQueryExecute(HibernateORMSession.java:657)
  at ortus.extension.orm.HibernateORMSession.executeQuery(HibernateORMSession.java:644)
  at ortus.extension.orm.functions.ORMExecuteQuery._call(ORMExecuteQuery.java:93)
  at ortus.extension.orm.functions.ORMExecuteQuery.call(ORMExecuteQuery.java:61)
  at ortus.extension.orm.functions.ORMExecuteQuery.invoke(ORMExecuteQuery.java:121)
  at lucee.runtime.functions.FunctionHandlerPool.invoke(FunctionHandlerPool.java:40)
  at lucee6.orm.ormtest2_cfm$cf$j.call(/lucee6/orm/ormtest2.cfm:12)
  at lucee.runtime.PageContextImpl._doInclude(PageContextImpl.java:1088)
  at lucee.runtime.PageContextImpl._doInclude(PageContextImpl.java:982)
  at lucee.runtime.listener.ModernAppListener._onRequest(ModernAppListener.java:213)
  at lucee.runtime.listener.MixedAppListener.onRequest(MixedAppListener.java:41)
  at lucee.runtime.PageContextImpl.execute(PageContextImpl.java:2767)
  at lucee.runtime.PageContextImpl._execute(PageContextImpl.java:2754)
  at lucee.runtime.PageContextImpl.executeCFML(PageContextImpl.java:2725)
  at lucee.runtime.engine.Request.exe(Request.java:45)
  at lucee.runtime.engine.CFMLEngineImpl._service(CFMLEngineImpl.java:1160)
  at lucee.runtime.engine.CFMLEngineImpl.serviceCFML(CFMLEngineImpl.java:1117)
  at lucee.loader.engine.CFMLEngineWrapper.serviceCFML(CFMLEngineWrapper.java:97)
  at lucee.loader.servlet.CFMLServlet.service(CFMLServlet.java:51)
  at javax.servlet.http.HttpServlet.service(HttpServlet.java:623)
  at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:199)
  at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:144)
  at org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:51)
  at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:168)
  at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:144)
  at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:168)
  at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:90)
  at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:481)
  at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:130)
  at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:93)
  at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:74)
  at org.apache.catalina.valves.RemoteIpValve.invoke(RemoteIpValve.java:761)
  at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:346)
  at org.apache.coyote.http11.Http11Processor.service(Http11Processor.java:388)
  at org.apache.coyote.AbstractProcessorLight.process(AbstractProcessorLight.java:63)
  at org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:936)
  at org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1791)
  at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:52)
  at org.apache.tomcat.util.threads.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1190)
  at org.apache.tomcat.util.threads.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:659)
  at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:63)
  at java.base/java.lang.Thread.run(Thread.java:840)
 Caused by: java.lang.ClassCastException: class java.lang.String cannot be cast to class java.math.BigDecimal (java.lang.String and java.math.BigDecimal are in module java.base of loader 'bootstrap')
  ... 66 more

Stepping through the Ortus ORM extension and the Lucee code, I’ve noticed that the hql parameter value when it is a BigDecimal, it gets converted from a BigDecimal to a String when it goes through this code in extension\src\main\java\ortus\extension\orm\HibernateORMSession.java

obj = HibernateCaster.toSQL( type, obj, isArray );

If the obj isn’t reassigned the bug goes away. That toSQL codes ends up going back to the Lucee code base via lucee.runtime.util.DBUtil.toSqlType. Which I think ends up here (I don’t have both code bases in my IDE and can’t step between the two projects very well):

Lucee\core\src\main\java\lucee\runtime\db\SQLCaster.java

And the hql big_integer type which was previously mapped to Types.DECIMAL has this suspect code:


			case Types.DECIMAL:
				return Caster.toString(Caster.toBigDecimal(value));

The change is fairly recent with a 2023 commit related to the improve and centralize BigDecimal creation from @micstriit .

Smoking gun? Red herring? I’m hoping somebody more familiar with the code base could step in or comment?

Thanks for bringing this up. Same thing affected me and I had to rewrite the code to force into numeric value. Hoping someone on the team can comment.

A jira issue has been created: [LDEV-5086] - Lucee