Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Release v2 #240

Merged
merged 107 commits into from
Jul 17, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
107 commits
Select commit Hold shift + click to select a range
b61ff5f
fix: Assume kotlin.Pair as a normal data class instead of an union
Chuckame Jan 21, 2024
79a132e
Merge pull request #174 from Chuckame/fix/kotlin-pair
Chuckame Jan 27, 2024
6a839b2
feat!: No more reflection and customizable logical types
Chuckame Jan 21, 2024
17eaf60
Merge pull request #175 from Chuckame/reflectionless
Chuckame Jan 27, 2024
dc27771
Merge pull request #177 from avro-kotlin/main
Chuckame Jan 27, 2024
c8492de
Generalize encoding/decoding tests (#168)
thake Jan 28, 2024
7696fe1
Merge pull request #179 from Chuckame/test/generalize-tests
Chuckame Jan 28, 2024
a485a3c
chore(build): Add spotless with ktlint + editorconfig
Chuckame Jan 28, 2024
4d59d4d
chore(spotless): Apply spotless
Chuckame Jan 28, 2024
24c9f2b
Merge pull request #180 from Chuckame/chore/formatting
Chuckame Jan 28, 2024
563c20f
feat: Fully support value classes and remove @AvroInline
Chuckame Jan 28, 2024
91dea8e
feat: Handle contextual map keys #114
Chuckame Jan 28, 2024
f6b186e
remove useless InlineDecoder as it is natively handled now
Chuckame Jan 29, 2024
b6f93e3
@AvroDefault & @ScalePrecision no more available for classes, as it i…
Chuckame Feb 8, 2024
1574c53
added assertThat test utils
Chuckame Feb 9, 2024
ebf6cc1
fixed decoding for map nullable values
Chuckame Feb 9, 2024
98cc0ae
remove useless line
Chuckame Apr 10, 2024
fdb53fc
apply spotless
Chuckame Apr 10, 2024
3cf493a
Merge pull request #183 from Chuckame/feat/value-classes
Chuckame Apr 10, 2024
f85e658
feat: Separate naming strategies (#178), remove AvroName[space] and a…
Chuckame Jan 28, 2024
237a3a1
feat: Add naming cache
Chuckame Feb 3, 2024
f0ec3bb
Merge pull request #182 from Chuckame/feat/separated-naming-strategy
Chuckame Apr 11, 2024
8ce714f
feat: Merge ScalePrecision to AvroDecimalLogicalType
Chuckame Apr 11, 2024
9a126af
Merge pull request #191 from Chuckame/feat/merge-decimal-ann
Chuckame Apr 11, 2024
cb35a3b
chore: Upgrade github actions and use standard gradle actions
Chuckame Apr 11, 2024
8aa48f3
Merge pull request #192 from Chuckame/gh-actions-update
Chuckame Apr 11, 2024
ea31c5d
chore: Add kover
Chuckame Apr 21, 2024
9e7ab01
deps: Remove snappy library
Chuckame Apr 21, 2024
29cc5e1
feat: revamp the schema generation
Chuckame Apr 21, 2024
69a369a
remove namingStrategy cache as there is a schema cache
Chuckame Apr 15, 2024
eee8a81
Merge pull request #190 from Chuckame/descriptor-visitor
Chuckame Apr 22, 2024
e3fdf92
feat: Handle char type
Chuckame Apr 21, 2024
bddc501
feat: New Avro entrypoint
Chuckame Apr 22, 2024
b3bf051
chore: remove redondant null checks
Chuckame Apr 22, 2024
b27b66d
chore: rename RecordNamingStrategy to TypeNamingStrategy
Chuckame Apr 22, 2024
85dc5c5
Merge pull request #186 from Chuckame/avro-api
Chuckame Apr 22, 2024
a4223ac
feat: Support everything at root level
Chuckame May 3, 2024
a6b4817
Merge pull request #202 from Chuckame/anything-root
Chuckame May 3, 2024
a4af252
feat!: Set @AvroEnumDefault directly to the enum value instead of the…
Chuckame May 3, 2024
bb5b035
feat!: Merge AvroJsonProp to AvroProp #201
Chuckame May 3, 2024
091f36c
Merge pull request #203 from Chuckame/enum-default
Chuckame May 3, 2024
325bd3f
Merge pull request #204 from Chuckame/unify-avro-json-prop
Chuckame May 3, 2024
a6df754
build: Explicit API mode to prevent exposing internal stuff
Chuckame May 3, 2024
e66ed3e
Merge pull request #205 from Chuckame/explicit-api-mode
Chuckame May 3, 2024
a16ca0c
deps: Upgrade kotlinx-serialization and kotlin
Chuckame May 9, 2024
0ec51a5
perf: Improve encoding performances with inlined union resolving
Chuckame May 7, 2024
a0d4272
chore: Add benchmark module
Chuckame May 9, 2024
a84209c
Merge pull request #208 from Chuckame/union-perf-improvement
Chuckame May 9, 2024
f3b2243
Merge pull request #209 from Chuckame/deps
Chuckame May 9, 2024
31b7889
docs: Improve README.md
Chuckame May 9, 2024
a030681
chore!: Stop supporting Timestamp class
Chuckame May 9, 2024
7c07046
chore! Remove custom schema as it is not supported in the serialization
Chuckame May 9, 2024
f3964d0
chore!: Internalize AvroLogicalType annotation, waiting for the new k…
Chuckame May 9, 2024
0c0725c
Merge pull request #210 from Chuckame/docs
Chuckame May 9, 2024
c8437a5
add sealed class in benchmark
Chuckame May 10, 2024
e8993e4
test: Support record with different field positions
Chuckame May 21, 2024
2d919da
test: Added nullable test case for nullable logical types
Chuckame May 11, 2024
7afa6ad
perf: Stop using tagger encoder
Chuckame May 11, 2024
be56714
perf: Stop using tagger decoder
Chuckame May 11, 2024
0ccd5d5
Add ExperimentalSerializationApi marker
Chuckame May 21, 2024
34e6393
feat!: No more kotlin-reflect for logical types
Chuckame May 23, 2024
7c90993
upgrade github workflows
Chuckame May 21, 2024
9cef510
Add gradle build scan
Chuckame May 23, 2024
8a99c60
docs: Add generic encoding note
Chuckame May 23, 2024
41dfff3
Merge pull request #214 from Chuckame/no-reflection
Chuckame May 23, 2024
b7f3b6b
perf: Direct decoding & encoding
Chuckame May 28, 2024
31306c1
Merge pull request #215 from Chuckame/direct-encoding
Chuckame May 29, 2024
c36c443
feat: Allow generating a release on a non-main branch
Chuckame May 29, 2024
2040d25
Merge pull request #217 from Chuckame/fix-release
Chuckame May 29, 2024
d3bc54b
feat: Allow adding props to a given type using value classes
Chuckame May 30, 2024
afc5fff
Merge pull request #219 from Chuckame/fix/props-value-class-element
Chuckame May 30, 2024
8c7b221
deps: Use non-RC version of kotlinx-serialization
Chuckame Jun 20, 2024
4804a9a
Merge pull request #221 from Chuckame/deps/non-rc-kotlinx-serialization
Chuckame Jun 20, 2024
a88f5cc
deps: Upgrade plugins, trying to fix publication failing
Chuckame Jun 20, 2024
9df5c94
Merge pull request #222 from Chuckame/deps/upgrade-versions
Chuckame Jun 20, 2024
78737b4
fix: Removed AvroNamespaceOverride as it was not fully implemented
Chuckame Jun 25, 2024
b6e52fc
feat: Implement BinaryFormat
Chuckame Jun 25, 2024
87c58ac
Add migration guide
Chuckame Jun 25, 2024
25f3f38
Merge pull request #224 from Chuckame/docs
Chuckame Jun 25, 2024
e5f5137
benchmark: Added apache avro lib generic<->binary benchmark
Chuckame Jun 25, 2024
7793a06
little refactor
Chuckame Jun 25, 2024
948643e
unifying union checks
Chuckame Jun 25, 2024
23814d2
Merge pull request #225 from Chuckame/improve-benchmark
Chuckame Jun 25, 2024
933fabb
fix docs
Chuckame Jun 26, 2024
78cf46f
Create dependabot.yml
Chuckame Jul 1, 2024
71a8e3c
Merge pull request #227 from avro-kotlin/Dependabot
Chuckame Jul 1, 2024
83544bd
Merge pull request #226 from Chuckame/docs
Chuckame Jul 1, 2024
228fcd1
handle nullable bytearrays and add null values in benchmark
Chuckame Jul 5, 2024
976dee7
Merge pull request #228 from Chuckame/docs
Chuckame Jul 5, 2024
5df37fb
feat: Add duration logical type
Chuckame Jul 10, 2024
ad17bb3
Merge pull request #233 from Chuckame/feat/duration-logical-type
Chuckame Jul 10, 2024
83934fc
fix: Only handle ByteArrays as bytes, and collection of Byte as array…
Chuckame Jul 10, 2024
489b45a
Merge pull request #234 from Chuckame/fix/remove-bytes
Chuckame Jul 10, 2024
01ccbfc
fix: No more automatic padding for fixed type
Chuckame Jul 10, 2024
34c32bb
Merge pull request #235 from Chuckame/feat/no-padding
Chuckame Jul 10, 2024
76d0134
fix: Update docs
Chuckame Jul 10, 2024
143d6a9
Merge pull request #237 from Chuckame/docs
Chuckame Jul 11, 2024
837ea2f
feat: Add @AvroStringable
Chuckame Jul 10, 2024
b0aec20
Merge pull request #236 from Chuckame/feat/stringable
Chuckame Jul 11, 2024
eb37957
feat: Remove AvroDecimal defaults
Chuckame Jul 11, 2024
eda4375
Merge pull request #238 from Chuckame/feat/decimal-default
Chuckame Jul 11, 2024
999a90c
feat: Add implicitEmptyCollections configuration
Chuckame Jul 14, 2024
967f5ed
Merge pull request #239 from Chuckame/implicit-empty-collections
Chuckame Jul 15, 2024
6a8dd19
Merge branch 'main' into main-v2
Chuckame Jul 16, 2024
84e88f4
fix documentation
Chuckame Jul 16, 2024
b64158f
benchmark: add simple use case
Chuckame Jul 16, 2024
f617d83
spotless + apiDump
Chuckame Jul 17, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
feat: Add @AvroStringable
  • Loading branch information
Chuckame committed Jul 11, 2024
commit 837ea2fe87f368ac893b331a383b01219b8f78cf
85 changes: 57 additions & 28 deletions README.md

Large diffs are not rendered by default.

30 changes: 12 additions & 18 deletions api/avro4k-core.api
Original file line number Diff line number Diff line change
Expand Up @@ -228,6 +228,13 @@ public final class com/github/avrokotlin/avro4k/AvroSingleObjectKt {
public static final fun encodeToByteArray (Lcom/github/avrokotlin/avro4k/AvroSingleObject;Lorg/apache/avro/Schema;Lkotlinx/serialization/SerializationStrategy;Ljava/lang/Object;)[B
}

public abstract interface annotation class com/github/avrokotlin/avro4k/AvroStringable : java/lang/annotation/Annotation {
}

public synthetic class com/github/avrokotlin/avro4k/AvroStringable$Impl : com/github/avrokotlin/avro4k/AvroStringable {
public fun <init> ()V
}

public abstract interface class com/github/avrokotlin/avro4k/BooleanValueDecoder {
public abstract fun decodeBoolean (Lcom/github/avrokotlin/avro4k/AvroDecoder;)Z
}
Expand Down Expand Up @@ -361,9 +368,11 @@ public abstract class com/github/avrokotlin/avro4k/serializer/AvroSerializer : c
}

public final class com/github/avrokotlin/avro4k/serializer/AvroSerializerKt {
public static final fun createSchema (Lcom/github/avrokotlin/avro4k/serializer/FoundElementAnnotation;)Lorg/apache/avro/Schema;
public static final fun getDecimal (Lcom/github/avrokotlin/avro4k/serializer/SchemaSupplierContext;)Lcom/github/avrokotlin/avro4k/serializer/FoundElementAnnotation;
public static final fun getFixed (Lcom/github/avrokotlin/avro4k/serializer/SchemaSupplierContext;)Lcom/github/avrokotlin/avro4k/serializer/FoundElementAnnotation;
public static final fun createSchema (Lcom/github/avrokotlin/avro4k/AvroFixed;Lcom/github/avrokotlin/avro4k/serializer/ElementLocation;)Lorg/apache/avro/Schema;
public static final fun createSchema (Lcom/github/avrokotlin/avro4k/AvroStringable;)Lorg/apache/avro/Schema;
public static final fun getDecimal (Lcom/github/avrokotlin/avro4k/serializer/ElementLocation;)Lcom/github/avrokotlin/avro4k/AvroDecimal;
public static final fun getFixed (Lcom/github/avrokotlin/avro4k/serializer/ElementLocation;)Lcom/github/avrokotlin/avro4k/AvroFixed;
public static final fun getStringable (Lcom/github/avrokotlin/avro4k/serializer/ElementLocation;)Lcom/github/avrokotlin/avro4k/AvroStringable;
}

public final class com/github/avrokotlin/avro4k/serializer/BigDecimalAsStringSerializer : com/github/avrokotlin/avro4k/serializer/AvroSerializer {
Expand Down Expand Up @@ -418,21 +427,6 @@ public final class com/github/avrokotlin/avro4k/serializer/ElementLocation {
public fun toString ()Ljava/lang/String;
}

public final class com/github/avrokotlin/avro4k/serializer/FoundElementAnnotation {
public fun <init> (Lkotlinx/serialization/descriptors/SerialDescriptor;ILjava/lang/annotation/Annotation;)V
public final fun component1 ()Lkotlinx/serialization/descriptors/SerialDescriptor;
public final fun component2 ()I
public final fun component3 ()Ljava/lang/annotation/Annotation;
public final fun copy (Lkotlinx/serialization/descriptors/SerialDescriptor;ILjava/lang/annotation/Annotation;)Lcom/github/avrokotlin/avro4k/serializer/FoundElementAnnotation;
public static synthetic fun copy$default (Lcom/github/avrokotlin/avro4k/serializer/FoundElementAnnotation;Lkotlinx/serialization/descriptors/SerialDescriptor;ILjava/lang/annotation/Annotation;ILjava/lang/Object;)Lcom/github/avrokotlin/avro4k/serializer/FoundElementAnnotation;
public fun equals (Ljava/lang/Object;)Z
public final fun getAnnotation ()Ljava/lang/annotation/Annotation;
public final fun getDescriptor ()Lkotlinx/serialization/descriptors/SerialDescriptor;
public final fun getElementIndex ()I
public fun hashCode ()I
public fun toString ()Ljava/lang/String;
}

public final class com/github/avrokotlin/avro4k/serializer/InstantSerializer : com/github/avrokotlin/avro4k/serializer/AvroSerializer {
public static final field INSTANCE Lcom/github/avrokotlin/avro4k/serializer/InstantSerializer;
public synthetic fun deserializeAvro (Lcom/github/avrokotlin/avro4k/AvroDecoder;)Ljava/lang/Object;
Expand Down
26 changes: 24 additions & 2 deletions src/main/kotlin/com/github/avrokotlin/avro4k/Annotations.kt
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
@file:OptIn(ExperimentalSerializationApi::class)

package com.github.avrokotlin.avro4k

import com.github.avrokotlin.avro4k.serializer.AvroSerializer
import com.github.avrokotlin.avro4k.serializer.BigDecimalSerializer
import kotlinx.serialization.ExperimentalSerializationApi
import kotlinx.serialization.SerialInfo
Expand Down Expand Up @@ -46,6 +45,29 @@ public annotation class AvroDoc(val value: String)
@Target(AnnotationTarget.PROPERTY, AnnotationTarget.CLASS)
public annotation class AvroAlias(vararg val value: String)

/**
* Sets the annotated property as a `string` type when inferring the class' schema. Takes precedence over any other schema modifier built-in annotation, except
* the fields referring to a custom serializer implementing [AvroSerializer] where you'll need to handle the annotation.
*
* If the given property is not string-able, the schema will be generated as a string, but it will fail at runtime during serialization or deserialization.
* You may need to provide a custom serializer for the given property to handle the string specifications.
*
* Only works with class properties for the following inferred schemas:
* - `string`
* - `boolean`
* - `int`
* - `long`
* - `float`
* - `double`
* - `bytes` & `fixed` (will take the fixed bytes as UTF-8 string, or custom toString/parse for logical types)
* - works also for all the nullable types of the above
* The rest will fail, except if your custom serializers handle the string type.
*/
@SerialInfo
@ExperimentalSerializationApi
@Target(AnnotationTarget.PROPERTY)
public annotation class AvroStringable

/**
* To be used with [BigDecimalSerializer] to specify the scale, precision, type and rounding mode of the decimal value.
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import com.github.avrokotlin.avro4k.serializer.SchemaSupplierContext
import com.github.avrokotlin.avro4k.serializer.SerialDescriptorWithAvroSchemaDelegate
import com.github.avrokotlin.avro4k.serializer.createSchema
import com.github.avrokotlin.avro4k.serializer.fixed
import com.github.avrokotlin.avro4k.serializer.stringable
import kotlinx.serialization.DeserializationStrategy
import kotlinx.serialization.InternalSerializationApi
import kotlinx.serialization.KSerializer
Expand Down Expand Up @@ -64,14 +65,17 @@ internal object SerializerLocatorMiddleware {

private val AvroStringSerialDescriptor: SerialDescriptor =
SerialDescriptorWithAvroSchemaDelegate(String.serializer().descriptor) { context ->
context.fixed?.createSchema() ?: Schema.create(Schema.Type.STRING)
context.inlinedElements.firstNotNullOfOrNull {
it.stringable?.createSchema() ?: it.fixed?.createSchema(it)
} ?: Schema.create(Schema.Type.STRING)
}

private object KotlinDurationSerializer : AvroSerializer<Duration>(Duration::class.qualifiedName!!) {
private const val MILLIS_PER_DAY = 1000 * 60 * 60 * 24

override fun getSchema(context: SchemaSupplierContext): Schema {
return AvroDurationSerializer.DURATION_SCHEMA
return context.inlinedElements.firstNotNullOfOrNull { it.stringable?.createSchema() }
?: AvroDurationSerializer.DURATION_SCHEMA
}

override fun serializeAvro(
Expand Down Expand Up @@ -128,7 +132,9 @@ private object KotlinDurationSerializer : AvroSerializer<Duration>(Duration::cla

private object AvroByteArraySerializer : AvroSerializer<ByteArray>(ByteArray::class.qualifiedName!!) {
override fun getSchema(context: SchemaSupplierContext): Schema {
return context.fixed?.createSchema() ?: Schema.create(Schema.Type.BYTES)
return context.inlinedElements.firstNotNullOfOrNull {
it.stringable?.createSchema() ?: it.fixed?.createSchema(it)
} ?: Schema.create(Schema.Type.BYTES)
}

override fun serializeAvro(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import kotlinx.serialization.encoding.CompositeEncoder
import kotlinx.serialization.modules.SerializersModule
import org.apache.avro.Schema
import org.apache.avro.generic.GenericFixed
import org.apache.avro.util.Utf8
import java.nio.ByteBuffer

internal class AvroValueDirectEncoder(
Expand Down Expand Up @@ -138,12 +139,14 @@ internal sealed class AbstractAvroDirectEncoder(
{ BadEncodedValueError(value, currentWriterSchema, Schema.Type.BYTES, Schema.Type.STRING, Schema.Type.FIXED) }
) {
when (it.type) {
Schema.Type.STRING,
Schema.Type.BYTES,
-> {
Schema.Type.BYTES -> {
{ binaryEncoder.writeBytes(value) }
}

Schema.Type.STRING -> {
{ binaryEncoder.writeString(Utf8(value.array())) }
}

Schema.Type.FIXED -> {
if (value.remaining() == it.fixedSize) {
{ binaryEncoder.writeFixed(value.array()) }
Expand All @@ -162,12 +165,14 @@ internal sealed class AbstractAvroDirectEncoder(
{ BadEncodedValueError(value, currentWriterSchema, Schema.Type.BYTES, Schema.Type.STRING, Schema.Type.FIXED) }
) {
when (it.type) {
Schema.Type.STRING,
Schema.Type.BYTES,
-> {
Schema.Type.BYTES -> {
{ binaryEncoder.writeBytes(value) }
}

Schema.Type.STRING -> {
{ binaryEncoder.writeString(Utf8(value)) }
}

Schema.Type.FIXED -> {
if (value.size == it.fixedSize) {
{ binaryEncoder.writeFixed(value) }
Expand All @@ -194,12 +199,14 @@ internal sealed class AbstractAvroDirectEncoder(
}
}

Schema.Type.STRING,
Schema.Type.BYTES,
-> {
Schema.Type.BYTES -> {
{ binaryEncoder.writeBytes(value.bytes()) }
}

Schema.Type.STRING -> {
{ binaryEncoder.writeString(Utf8(value.bytes())) }
}

else -> null
}
}
Expand All @@ -217,12 +224,14 @@ internal sealed class AbstractAvroDirectEncoder(
null
}

Schema.Type.STRING,
Schema.Type.BYTES,
-> {
Schema.Type.BYTES -> {
{ binaryEncoder.writeBytes(value) }
}

Schema.Type.STRING -> {
{ binaryEncoder.writeString(Utf8(value)) }
}

else -> null
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import com.github.avrokotlin.avro4k.internal.jsonNode
import com.github.avrokotlin.avro4k.internal.nonNullSerialName
import com.github.avrokotlin.avro4k.internal.nullable
import com.github.avrokotlin.avro4k.serializer.AvroSchemaSupplier
import com.github.avrokotlin.avro4k.serializer.stringable
import kotlinx.serialization.descriptors.PolymorphicKind
import kotlinx.serialization.descriptors.PrimitiveKind
import kotlinx.serialization.descriptors.SerialDescriptor
Expand Down Expand Up @@ -84,9 +85,17 @@ internal class ValueVisitor internal constructor(
override fun visitValue(descriptor: SerialDescriptor) {
val finalDescriptor = SerializerLocatorMiddleware.apply(unwrapNullable(descriptor))

(finalDescriptor.nonNullOriginal as? AvroSchemaSupplier)
?.getSchema(context)?.let { setSchema(it) }
?: super.visitValue(finalDescriptor)
if (finalDescriptor is AvroSchemaSupplier) {
setSchema(finalDescriptor.getSchema(context))
return
}

if (context.inlinedElements.any { it.stringable != null }) {
setSchema(Schema.create(Schema.Type.STRING))
return
}

super.visitValue(finalDescriptor)
}

private fun unwrapNullable(descriptor: SerialDescriptor): SerialDescriptor {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,8 +44,8 @@ public data class AvroDuration(
if (days != 0u) {
append("${days}D")
}
append("T")
if (millis != 0u) {
append("T")
append(millis / 1000u)
val millisPart = millis % 1000u
if (millisPart != 0u) {
Expand Down Expand Up @@ -89,12 +89,20 @@ public data class AvroDuration(
val match = PATTERN.matchEntire(value) ?: return null
val (years, months, weeks, days, hours, minutes, seconds, millis) = match.destructured
return AvroDuration(
months = years.toUInt() * 12u + months.toUInt(),
days = weeks.toUInt() * 7u + days.toUInt(),
millis = hours.toUInt() * 60u * 60u * 1000u + minutes.toUInt() * 60u * 1000u + seconds.toUInt() * 1000u + millis.toUInt()
months = years * 12u + months.toUIntOrZero(),
days = weeks * 7u + days.toUIntOrZero(),
millis = hours * 60u * 60u * 1000u + minutes * 60u * 1000u + seconds * 1000u + millis.toUIntOrZero()
)
}

private operator fun String.times(other: UInt): UInt {
return toUIntOrNull()?.times(other) ?: 0u
}

private fun String.toUIntOrZero(): UInt {
return toUIntOrNull() ?: 0u
}

@JvmStatic
public fun parse(value: String): AvroDuration {
return tryParse(value) ?: throw AvroDurationParseException(value)
Expand All @@ -114,7 +122,9 @@ internal object AvroDurationSerializer : AvroSerializer<AvroDuration>(AvroDurati
}

override fun getSchema(context: SchemaSupplierContext): Schema {
return DURATION_SCHEMA
return context.inlinedElements.firstNotNullOfOrNull {
it.stringable?.createSchema()
} ?: DURATION_SCHEMA
}

override fun serializeAvro(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import com.github.avrokotlin.avro4k.AvroDecimal
import com.github.avrokotlin.avro4k.AvroDecoder
import com.github.avrokotlin.avro4k.AvroEncoder
import com.github.avrokotlin.avro4k.AvroFixed
import com.github.avrokotlin.avro4k.AvroStringable
import com.github.avrokotlin.avro4k.internal.findElementAnnotation
import com.github.avrokotlin.avro4k.internal.namespace
import kotlinx.serialization.ExperimentalSerializationApi
Expand Down Expand Up @@ -100,57 +101,53 @@ public interface SchemaSupplierContext {
}

/**
* Search for the first annotation of type [T] in the [SchemaSupplierContext.inlinedElements].
*
* The top-est matching annotation is returned, that means the one that is closest to the class element.
* Search for the first annotation of type [T] in the given [ElementLocation].
*/
@ExperimentalSerializationApi
public inline fun <reified T : Annotation> SchemaSupplierContext.findAnnotation(): FoundElementAnnotation<T>? {
return inlinedElements.firstNotNullOfOrNull { elementLocation ->
elementLocation.descriptor.findElementAnnotation<T>(elementLocation.elementIndex)?.let {
FoundElementAnnotation(elementLocation.descriptor, elementLocation.elementIndex, it)
}
}
public inline fun <reified T : Annotation> ElementLocation.findAnnotation(): T? {
return descriptor.findElementAnnotation(elementIndex)
}

/**
* Shorthand for [findAnnotation] with [AvroDecimal] as it is a built-in annotation.
*/
@ExperimentalSerializationApi
public val SchemaSupplierContext.decimal: FoundElementAnnotation<AvroDecimal>?
public val ElementLocation.decimal: AvroDecimal?
get() = findAnnotation()

/**
* Shorthand for [findAnnotation] with [AvroFixed] as it is a built-in annotation.
* Shorthand for [findAnnotation] with [AvroStringable] as it is a built-in annotation.
*/
@ExperimentalSerializationApi
public val SchemaSupplierContext.fixed: FoundElementAnnotation<AvroFixed>?
public val ElementLocation.stringable: AvroStringable?
get() = findAnnotation()

/**
* Creates a fixed schema from the [AvroFixed] annotation.
* Creates a string schema from the [AvroStringable] annotation.
*/
@ExperimentalSerializationApi
public fun FoundElementAnnotation<AvroFixed>.createSchema(): Schema = Schema.createFixed(descriptor.getElementName(elementIndex), null, descriptor.namespace, annotation.size)
public fun AvroStringable.createSchema(): Schema = Schema.create(Schema.Type.STRING)

/**
* Shorthand for [findAnnotation] with [AvroFixed] as it is a built-in annotation.
*/
@ExperimentalSerializationApi
public data class ElementLocation
@PublishedApi
internal constructor(
val descriptor: SerialDescriptor,
val elementIndex: Int,
)
public val ElementLocation.fixed: AvroFixed?
get() = findAnnotation()

/**
* Represents a found annotation on an element, provided by [findAnnotation].
* Creates a fixed schema from the [AvroFixed] annotation.
*/
@ExperimentalSerializationApi
public data class FoundElementAnnotation<T : Annotation>
public fun AvroFixed.createSchema(elementLocation: ElementLocation): Schema =
Schema.createFixed(elementLocation.descriptor.getElementName(elementLocation.elementIndex), null, elementLocation.descriptor.namespace, size)

@ExperimentalSerializationApi
public data class ElementLocation
@PublishedApi
internal constructor(
val descriptor: SerialDescriptor,
val elementIndex: Int,
val annotation: T,
)

internal fun interface AvroSchemaSupplier {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -185,10 +185,10 @@ private val defaultAnnotation = AvroDecimal()

public object BigDecimalSerializer : AvroSerializer<BigDecimal>(BigDecimal::class.qualifiedName!!) {
override fun getSchema(context: SchemaSupplierContext): Schema {
val decimalAnnotation = context.decimal?.annotation ?: defaultAnnotation
val schema = context.fixed?.createSchema() ?: Schema.create(Schema.Type.BYTES)
decimalAnnotation.logicalType.addToSchema(schema)
return schema
val logicalType = (context.inlinedElements.firstNotNullOfOrNull { it.decimal } ?: defaultAnnotation).logicalType
return context.inlinedElements.firstNotNullOfOrNull {
it.stringable?.createSchema() ?: it.fixed?.createSchema(it)?.copy(logicalType = logicalType)
} ?: Schema.create(Schema.Type.BYTES).copy(logicalType = logicalType)
}

override fun serializeAvro(
Expand Down
Loading