From 56393493db7248b5307a842aea39092902ca4b9e Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 9 Dec 2024 20:04:41 -0800 Subject: [PATCH 1/4] Bump actions/checkout from 4.1.6 to 4.2.2 (#532) --- .github/workflows/dep_build_v2.yml | 2 +- .github/workflows/dep_build_v3.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/dep_build_v2.yml b/.github/workflows/dep_build_v2.yml index ff1e9d8fb..dde49d619 100644 --- a/.github/workflows/dep_build_v2.yml +++ b/.github/workflows/dep_build_v2.yml @@ -19,7 +19,7 @@ jobs: env: JAVA_OPTS: "-XX:+TieredCompilation -XX:TieredStopAtLevel=1" steps: - - uses: actions/checkout@a5ac7e51b41094c92402da3b24376905380afc29 # v4.1.6 + - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 - name: Set up JDK uses: actions/setup-java@99b8673ff64fbf99d8d325f52d9a5bdedb8483e9 # v4.2.1 with: diff --git a/.github/workflows/dep_build_v3.yml b/.github/workflows/dep_build_v3.yml index 4f3978b09..3caf3004c 100644 --- a/.github/workflows/dep_build_v3.yml +++ b/.github/workflows/dep_build_v3.yml @@ -19,7 +19,7 @@ jobs: env: JAVA_OPTS: "-XX:+TieredCompilation -XX:TieredStopAtLevel=1" steps: - - uses: actions/checkout@a5ac7e51b41094c92402da3b24376905380afc29 # v4.1.6 + - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 with: ref: master - name: Set up JDK From b72deb0ab109e67cc0008e96f332a0f8faa77970 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 9 Dec 2024 20:05:12 -0800 Subject: [PATCH 2/4] Bump codecov/codecov-action from 5.0.7 to 5.1.1 (#533) --- .github/workflows/main.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index e4712a117..2fd790136 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -62,7 +62,7 @@ jobs: run: ./mvnw -B -q -ff -ntp test - name: Publish code coverage if: ${{ matrix.release_build && github.event_name != 'pull_request' }} - uses: codecov/codecov-action@015f24e6818733317a2da2edd6290ab26238649a # v5.0.7 + uses: codecov/codecov-action@7f8b4b4bde536c465e797be725718b88c5d95e0e # v5.1.1 with: token: ${{ secrets.CODECOV_TOKEN }} file: ./target/site/jacoco/jacoco.xml From ad37021090fd601f3be3c49b8e4690574ae82c67 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 9 Dec 2024 20:06:13 -0800 Subject: [PATCH 3/4] Bump actions/setup-java from 4.2.1 to 4.5.0 (#534) --- .github/workflows/dep_build_v2.yml | 2 +- .github/workflows/dep_build_v3.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/dep_build_v2.yml b/.github/workflows/dep_build_v2.yml index dde49d619..77b295b95 100644 --- a/.github/workflows/dep_build_v2.yml +++ b/.github/workflows/dep_build_v2.yml @@ -21,7 +21,7 @@ jobs: steps: - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 - name: Set up JDK - uses: actions/setup-java@99b8673ff64fbf99d8d325f52d9a5bdedb8483e9 # v4.2.1 + uses: actions/setup-java@8df1039502a15bceb9433410b1a100fbe190c53b # v4.5.0 with: distribution: 'temurin' java-version: ${{ matrix.java_version }} diff --git a/.github/workflows/dep_build_v3.yml b/.github/workflows/dep_build_v3.yml index 3caf3004c..5e63e943b 100644 --- a/.github/workflows/dep_build_v3.yml +++ b/.github/workflows/dep_build_v3.yml @@ -23,7 +23,7 @@ jobs: with: ref: master - name: Set up JDK - uses: actions/setup-java@99b8673ff64fbf99d8d325f52d9a5bdedb8483e9 # v4.2.1 + uses: actions/setup-java@8df1039502a15bceb9433410b1a100fbe190c53b # v4.5.0 with: distribution: 'temurin' java-version: ${{ matrix.java_version }} From 10fc1ba4b4e243d277c72833d4df8478ecd875bc Mon Sep 17 00:00:00 2001 From: Michal Foksa Date: Tue, 24 Dec 2024 02:29:42 +0100 Subject: [PATCH 4/4] Add Logical Type support for `java.util.UUID` (#536) --- avro/README.md | 40 ++++++++-------- .../avro/schema/AvroSchemaHelper.java | 2 +- .../dataformat/avro/schema/StringVisitor.java | 6 +-- .../dataformat/avro/schema/UUIDVisitor.java | 43 +++++++++++++++++ .../avro/schema/VisitorFormatWrapperImpl.java | 6 +++ .../UUIDVisitor_builtAvroSchemaTest.java | 46 +++++++++++++++++++ release-notes/CREDITS-2.x | 2 + release-notes/VERSION-2.x | 5 ++ 8 files changed, 125 insertions(+), 25 deletions(-) create mode 100644 avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/UUIDVisitor.java create mode 100644 avro/src/test/java/com/fasterxml/jackson/dataformat/avro/schema/UUIDVisitor_builtAvroSchemaTest.java diff --git a/avro/README.md b/avro/README.md index d01db46a6..6edcab9ec 100644 --- a/avro/README.md +++ b/avro/README.md @@ -114,32 +114,34 @@ and that's about it, for now. ## Avro Logical Types -Following is an extract from [Logical Types](http://avro.apache.org/docs/current/specification/_print/#logical-types) paragraph in -Avro schema specification: +The following is an excerpt from the [Logical Types](https://avro.apache.org/docs/1.11.1/specification/#logical-types) section of +the Avro schema specification: + > A logical type is an Avro primitive or complex type with extra attributes to represent a derived type. The attribute -> `logicalType` is always be present for a logical type, and is a string with the name of one of the logical types -> defined by Avro specification. +> `logicalType` must always be present for a logical type, and is a string with the name of one of the logical types +> listed later in this section. Other attributes may be defined for particular logical types. + +Logical types are supported for a limited set of `java.time` classes and for 'java.util.UUID'. See the table below for more details. -Generation of logical types for limited set of `java.time` classes is supported at the moment. See a table bellow. +### Mapping to Logical Types -### Mapping to Logical Type +Mapping to Avro type and logical type involves these steps: -Mapping to Avro type and logical type works in few steps: -1. Serializer for particular Java type (or class) determines a Jackson type where the Java type will be serialized into. -2. `AvroSchemaGenerator` determines corresponding Avro type for that Jackson type. -2. If logical type generation is enabled, then `logicalType` is determined for the above combination of Java type and - Avro type. +1. The serializer for a Java type identifies the Jackson type it will serialize into. +2. The `AvroSchemaGenerator` maps that Jackson type to the corresponding Avro type. +3. `logicalType` value is combination of Java type and Jackson type. #### Java type to Avro Logical Type mapping -| Java type | Serialization type | Generated Avro schema with Avro type and logical type -| ----------------------------- | ------------------ | ----------------------------------------------------- -| `java.time.OffsetDateTime` | NumberType.LONG | `{"type": "long", "logicalType": "timestamp-millis"}` -| `java.time.ZonedDateTime` | NumberType.LONG | `{"type": "long", "logicalType": "timestamp-millis"}` -| `java.time.Instant` | NumberType.LONG | `{"type": "long", "logicalType": "timestamp-millis"}` -| `java.time.LocalDate` | NumberType.INT | `{"type": "int", "logicalType": "date"}` -| `java.time.LocalTime` | NumberType.INT | `{"type": "int", "logicalType": "time-millis"}` -| `java.time.LocalDateTime` | NumberType.LONG | `{"type": "long", "logicalType": "local-timestamp-millis"}` +| Java type | Jackson type | Generated Avro schema with logical type | +|----------------------------|-----------------|---------------------------------------------------------------------------------------------------| +| `java.time.OffsetDateTime` | NumberType.LONG | `{"type": "long", "logicalType": "timestamp-millis"}` | +| `java.time.ZonedDateTime` | NumberType.LONG | `{"type": "long", "logicalType": "timestamp-millis"}` | +| `java.time.Instant` | NumberType.LONG | `{"type": "long", "logicalType": "timestamp-millis"}` | +| `java.time.LocalDate` | NumberType.INT | `{"type": "int", "logicalType": "date"}` | +| `java.time.LocalTime` | NumberType.INT | `{"type": "int", "logicalType": "time-millis"}` | +| `java.time.LocalDateTime` | NumberType.LONG | `{"type": "long", "logicalType": "local-timestamp-millis"}` | +| `java.util.UUID` (2.19+) | | `{"type": "fixed", "name": "UUID", "namespace": "java.util", "size": 16, "logicalType" : "uuid"}` | _Provided Avro logical type generation is enabled._ diff --git a/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/AvroSchemaHelper.java b/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/AvroSchemaHelper.java index 00f0592f6..99aa79937 100644 --- a/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/AvroSchemaHelper.java +++ b/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/AvroSchemaHelper.java @@ -286,7 +286,7 @@ public static Schema createEnumSchema(BeanDescription bean, List values) * @since 2.11 */ public static Schema createUUIDSchema() { - return Schema.createFixed("UUID", "", "java.util", 16); + return Schema.createFixed("UUID", null, "java.util", 16); } /** diff --git a/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/StringVisitor.java b/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/StringVisitor.java index b8f3f5147..4aabb9dca 100644 --- a/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/StringVisitor.java +++ b/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/StringVisitor.java @@ -39,11 +39,7 @@ public Schema builtAvroSchema() { // should we construct JavaType for `Character.class` in case of primitive or... ? return AvroSchemaHelper.numericAvroSchema(NumberType.INT, _type); } - // [dataformats-binary#179]: need special help with UUIDs, to coerce into Binary - // (could actually be - if (_type.hasRawClass(java.util.UUID.class)) { - return AvroSchemaHelper.createUUIDSchema(); - } + BeanDescription bean = _provider.getConfig().introspectClassAnnotations(_type); Schema schema = Schema.create(Schema.Type.STRING); // Stringable classes need to include the type diff --git a/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/UUIDVisitor.java b/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/UUIDVisitor.java new file mode 100644 index 000000000..914fb954e --- /dev/null +++ b/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/UUIDVisitor.java @@ -0,0 +1,43 @@ +package com.fasterxml.jackson.dataformat.avro.schema; + +import java.util.Set; + +import com.fasterxml.jackson.databind.jsonFormatVisitors.JsonStringFormatVisitor; +import com.fasterxml.jackson.databind.jsonFormatVisitors.JsonValueFormat; + +import org.apache.avro.LogicalTypes; +import org.apache.avro.Schema; + +/** + * Visitor for {@link java.util.UUID} type. When it is created with logicalTypesEnabled enabled, + * Avro schema is created with logical type uuid. + * + * @since 2.19 + */ +public class UUIDVisitor extends JsonStringFormatVisitor.Base + implements SchemaBuilder { + protected boolean _logicalTypesEnabled = false; + + + public UUIDVisitor(boolean logicalTypesEnabled) { + _logicalTypesEnabled = logicalTypesEnabled; + } + + @Override + public void format(JsonValueFormat format) { + // Ideally, we'd recognize UUIDs, Dates etc if need be, here... + } + + @Override + public void enumTypes(Set enums) { + // Do nothing + } + + @Override + public Schema builtAvroSchema() { + // [dataformats-binary#179]: need special help with UUIDs, to coerce into Binary + // (could actually be + Schema schema = AvroSchemaHelper.createUUIDSchema(); + return this._logicalTypesEnabled ? LogicalTypes.uuid().addToSchema(schema) : schema; + } +} diff --git a/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/VisitorFormatWrapperImpl.java b/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/VisitorFormatWrapperImpl.java index a2b9d6edc..247d6b816 100644 --- a/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/VisitorFormatWrapperImpl.java +++ b/avro/src/main/java/com/fasterxml/jackson/dataformat/avro/schema/VisitorFormatWrapperImpl.java @@ -214,6 +214,12 @@ public JsonStringFormatVisitor expectStringFormat(JavaType type) return v; } + if (type.hasRawClass(java.util.UUID.class)) { + UUIDVisitor v = new UUIDVisitor(this._logicalTypesEnabled); + _builder = v; + return v; + } + StringVisitor v = new StringVisitor(_provider, type); _builder = v; return v; diff --git a/avro/src/test/java/com/fasterxml/jackson/dataformat/avro/schema/UUIDVisitor_builtAvroSchemaTest.java b/avro/src/test/java/com/fasterxml/jackson/dataformat/avro/schema/UUIDVisitor_builtAvroSchemaTest.java new file mode 100644 index 000000000..847fb399a --- /dev/null +++ b/avro/src/test/java/com/fasterxml/jackson/dataformat/avro/schema/UUIDVisitor_builtAvroSchemaTest.java @@ -0,0 +1,46 @@ +package com.fasterxml.jackson.dataformat.avro.schema; + +import org.junit.Test; + +import org.apache.avro.LogicalType; +import org.apache.avro.Schema; + +import static org.assertj.core.api.Assertions.assertThat; + +public class UUIDVisitor_builtAvroSchemaTest { + + @Test + public void testLogicalTypesDisabled() { + // GIVEN + boolean logicalTypesEnabled = false; + UUIDVisitor uuidVisitor = new UUIDVisitor(logicalTypesEnabled); + + // WHEN + Schema actualSchema = uuidVisitor.builtAvroSchema(); + + // THEN + assertThat(actualSchema.getType()).isEqualTo(Schema.Type.FIXED); + assertThat(actualSchema.getFixedSize()).isEqualTo(16); + assertThat(actualSchema.getName()).isEqualTo("UUID"); + assertThat(actualSchema.getNamespace()).isEqualTo("java.util"); + assertThat(actualSchema.getProp(LogicalType.LOGICAL_TYPE_PROP)).isNull(); + } + + @Test + public void testLogicalTypesEnabled() { + // GIVEN + boolean logicalTypesEnabled = true; + UUIDVisitor uuidVisitor = new UUIDVisitor(logicalTypesEnabled); + + // WHEN + Schema actualSchema = uuidVisitor.builtAvroSchema(); + + // THEN + assertThat(actualSchema.getType()).isEqualTo(Schema.Type.FIXED); + assertThat(actualSchema.getFixedSize()).isEqualTo(16); + assertThat(actualSchema.getName()).isEqualTo("UUID"); + assertThat(actualSchema.getNamespace()).isEqualTo("java.util"); + assertThat(actualSchema.getProp(LogicalType.LOGICAL_TYPE_PROP)).isEqualTo("uuid"); + } + +} diff --git a/release-notes/CREDITS-2.x b/release-notes/CREDITS-2.x index 5c7ddb2ea..d6f98acae 100644 --- a/release-notes/CREDITS-2.x +++ b/release-notes/CREDITS-2.x @@ -225,6 +225,8 @@ Michal Foksa (MichalFoksa@github) * Contributed #494: Avro Schema generation: allow mapping Java Enum properties to Avro String values (2.18.0) +* Contributed #536: (avro) Add Logical Type support for `java.util.UUID` + (2.19.0) Hunter Herman (hherman1@github) diff --git a/release-notes/VERSION-2.x b/release-notes/VERSION-2.x index 28e71468e..3b808b834 100644 --- a/release-notes/VERSION-2.x +++ b/release-notes/VERSION-2.x @@ -14,6 +14,11 @@ Active maintainers: === Releases === ------------------------------------------------------------------------ +2.19.0 (not yet released) + +#536: (avro) Add Logical Type support for `java.util.UUID` + (contributed by Michal F) + 2.18.2 (27-Nov-2024) No changes since 2.18.1