From 887ac3cc1d0173cc713beb65cfb91e788e827c20 Mon Sep 17 00:00:00 2001 From: Daniel Fagerstrom Date: Mon, 2 Dec 2024 02:29:03 +0100 Subject: [PATCH] Enable json values for @AvroProp. (#855) Co-authored-by: Daniel Fagerstrom --- .../scala/com/sksamuel/avro4s/annotations.scala | 5 +++-- .../sksamuel/avro4s/typeutils/Annotations.scala | 3 ++- .../resources/props_annotation_json_field.json | 15 +++++++++++++++ .../avro4s/schema/AvroPropSchemaTest.scala | 8 ++++++++ 4 files changed, 28 insertions(+), 3 deletions(-) create mode 100644 avro4s-core/src/test/resources/props_annotation_json_field.json diff --git a/avro4s-core/src/main/scala/com/sksamuel/avro4s/annotations.scala b/avro4s-core/src/main/scala/com/sksamuel/avro4s/annotations.scala index 2c61b453..90cc92c3 100644 --- a/avro4s-core/src/main/scala/com/sksamuel/avro4s/annotations.scala +++ b/avro4s-core/src/main/scala/com/sksamuel/avro4s/annotations.scala @@ -1,5 +1,6 @@ package com.sksamuel.avro4s +import com.fasterxml.jackson.databind.JsonNode import scala.annotation.StaticAnnotation case class AvroAlias(override val alias: String) extends AvroAliasable @@ -79,11 +80,11 @@ trait AvroNamespaceable extends AvroFieldReflection { val namespace: String } -case class AvroProp(override val key: String, override val value:String) extends AvroProperty +case class AvroProp(override val key: String, override val value: String | JsonNode) extends AvroProperty trait AvroProperty extends AvroFieldReflection { val key: String - val value: String + val value: String | JsonNode } /** diff --git a/avro4s-core/src/main/scala/com/sksamuel/avro4s/typeutils/Annotations.scala b/avro4s-core/src/main/scala/com/sksamuel/avro4s/typeutils/Annotations.scala index 9aed98db..90a101cd 100644 --- a/avro4s-core/src/main/scala/com/sksamuel/avro4s/typeutils/Annotations.scala +++ b/avro4s-core/src/main/scala/com/sksamuel/avro4s/typeutils/Annotations.scala @@ -1,5 +1,6 @@ package com.sksamuel.avro4s.typeutils +import com.fasterxml.jackson.databind.JsonNode import com.sksamuel.avro4s.{AvroAliasable, AvroDoc, AvroDocumentable, AvroErasedName, AvroError, AvroFixed, AvroName, AvroNameable, AvroNamespace, AvroNoDefault, AvroProp, AvroProperty, AvroSortPriority, AvroTransient, AvroUnionPosition} import magnolia1.{CaseClass, TypeInfo} @@ -18,7 +19,7 @@ class Annotations(annos: Seq[Any], inheritedAnnos: Seq[Any] = Nil) { case t: AvroAliasable => t.alias }.filterNot(_.trim.isEmpty) - def props: Map[String, String] = annos.collect { + def props: Map[String, String | JsonNode] = annos.collect { case t: AvroProperty => (t.key, t.value) }.toMap diff --git a/avro4s-core/src/test/resources/props_annotation_json_field.json b/avro4s-core/src/test/resources/props_annotation_json_field.json new file mode 100644 index 00000000..bca3a018 --- /dev/null +++ b/avro4s-core/src/test/resources/props_annotation_json_field.json @@ -0,0 +1,15 @@ +{ + "type": "record", + "name": "Annotated", + "namespace": "com.sksamuel.avro4s.schema.AvroPropSchemaTest", + "fields": [ + { + "name": "str", + "type": "string", + "terms": [ + "foo", + "bar" + ] + } + ] +} \ No newline at end of file diff --git a/avro4s-core/src/test/scala/com/sksamuel/avro4s/schema/AvroPropSchemaTest.scala b/avro4s-core/src/test/scala/com/sksamuel/avro4s/schema/AvroPropSchemaTest.scala index c5241b6c..abdf7092 100644 --- a/avro4s-core/src/test/scala/com/sksamuel/avro4s/schema/AvroPropSchemaTest.scala +++ b/avro4s-core/src/test/scala/com/sksamuel/avro4s/schema/AvroPropSchemaTest.scala @@ -1,5 +1,6 @@ package com.sksamuel.avro4s.schema +import com.fasterxml.jackson.databind.{JsonNode, ObjectMapper} import com.sksamuel.avro4s.{AvroProp, AvroSchema} import org.scalatest.matchers.should.Matchers import org.scalatest.wordspec.AnyWordSpec @@ -19,6 +20,13 @@ class AvroPropSchemaTest extends AnyWordSpec with Matchers { val schema = AvroSchema[Annotated] schema.toString(true) shouldBe expected.toString(true) } + "support json prop annotation on field" in { + val jsonArray = (new ObjectMapper()).createArrayNode().add("foo").add("bar") + case class Annotated(@AvroProp("terms", jsonArray) str: String) + val expected = new org.apache.avro.Schema.Parser().parse(getClass.getResourceAsStream("/props_annotation_json_field.json")) + val schema = AvroSchema[Annotated] + schema.toString(true) shouldBe expected.toString(true) + } // "support props annotations on scala enums" in { // case class Annotated(@AvroProp("cold", "play") colours: Colours.Value) // val expected = new org.apache.avro.Schema.Parser().parse(getClass.getResourceAsStream("/props_annotation_scala_enum.json"))