近日接到与其他系统对接的需求,已完成开发工作,进行联调测试时出现了奇怪的问题。部署完成后,成功连接到其他系统数据库,但无法读取数据。在创建完成query后,执行query.getResultList()方法报错。
报错的sql为:SELECT ID FROM A#B.C WHERE LAST_UPDATE_TIME>时间。
其中A#B为数据库用户名,C为表名。
解释下场景:用户名A#B是要对接系统的数据库名,因安全问题,并不能直接供我方使用,对接系统提供了用户名A#F用户,仅授权了 A#B.C的查询权限。程序是通过 A#F用户链接,读取C表时,需通过 A#B.C的方式读取。
该条sql在ORACLE 数据库开发工具中是能成功执行,并返回结果,但是在程序中无法执行。
报错日志如下:
java.lang.NullPointerException
at org.eclipse.persistence.internal.databaseaccess.DatabaseCall.translate(DatabaseCall.java:978)
at org.eclipse.persistence.internal.queries.DatasourceCallQueryMechanism.executeCall(DatasourceCallQueryMechanism.java:206)
at org.eclipse.persistence.internal.queries.DatasourceCallQueryMechanism.executeCall(DatasourceCallQueryMechanism.java:193)
排查过程:
1、经过代码分析,发现报错的日志为jar包中的代码,不是编写的业务代码报错。
2、经过jar包源码分析,发现是jar包将报错的内容处理了,实际的报错信息没有打印出来。
3、在无法调整jar包的情况下,只能将报错的代码进行调试。重点查看conn、query、在vm执行时的变量。
4、最终发现是因为外部系统的用户名包含“#”符合,该符合在jpa框架中是占位符,框架在解析该条语句时,会解析为 "SELECT ID FROM ? WHERE LAST_UPDATE_TIME>时间"。导致程序报错。
解决方案:
1、建立dblink。使用SELECT ID FROM C@DBLINK WHERE LAST_UPDATE_TIME>时间的方式进行查询。
2、调整用户名。将包含特殊字符的用户名调整为不含特殊字符的用户名。例如:A#C调整为 A_C。