Jacob Stampe Mikkelsen created AVRO-3989:
--------------------------------------------
Summary: Schema conversion for logical types ignored when type is
part of union
Key: AVRO-3989
URL: https://issues.apache.org/jira/browse/AVRO-3989
Project: Apache Avro
Issue Type: Bug
Components: java
Affects Versions: 1.11.3
Reporter: Jacob Stampe Mikkelsen
I have a schema with an optional timestamp which is to be rendered as millis
and the generated Java class fails to get the conversions array and method
added, which means that at runtime I get a ClassCastException when an "Instant"
object is cast to a "Long" because there was no registered converter:
The element with an optional timestamp in my schema:
{{{}}
{{ "name": "DecisionStartTime",}}
{{ "type": [}}
{{ "null",}}
{{ {}}
{{ "type": "long",}}
{{ "logicalType": "timestamp-millis"}}
{{ }}}
{{ ],}}
{{ "default": null}}
{{}}}
The problem seems to be here:
[https://github.com/apache/avro/blame/5a60b5c43fe9c3bb40f4a1303913fdb3598bd4a5/lang/java/compiler/src/main/velocity/org/apache/avro/compiler/specific/templates/java/classic/record.vm#L198]
Which only renders the conversion part if any of the fields has logicalType set
and the actual method:
[https://github.com/apache/avro/blob/5a60b5c43fe9c3bb40f4a1303913fdb3598bd4a5/lang/java/compiler/src/main/java/org/apache/avro/compiler/specific/SpecificCompiler.java#L976C3-L983C4]
Does not take into account that a field which is a union does not have a
logical type, however one of the union types might.
*Proposed solution* - replace method
{{ public boolean hasLogicalTypeField(Schema schema) {}}
{{ for (Schema.Field field : schema.getFields()) {}}
{{ if (field.schema().getLogicalType() != null) {}}
{{ return true;}}
{{ }}}
{{ }}}
{{ return false;}}
{{ }}}
With a method like this:
{{ public boolean hasLogicalTypeField(Schema schema) {}}
{{ for (Schema.Field field : schema.getFields()) {}}
{{ if (field.schema().getLogicalType() != null) {}}
{{ return true;}}
{{ } else if (field.schema().type == Type.UNION) {}}
{{ for (Schema type : field.schema().getTypes()) {}}
{{ if (type.getLogicalType() != null) {}}
{{ return true;}}
{{ }}}
{{ }}}
{{ }}}
{{ }}}
{{ return false;}}
{{ }}}
The conversionInstance() method might also need tweaking in order to identify
that the given type has a logical type as part of the union.
--
This message was sent by Atlassian Jira
(v8.20.10#820010)