-
Notifications
You must be signed in to change notification settings - Fork 121
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
[Feature Request]: Schema Registry Samples is Missing an Avro Generic Example #532
Comments
For the sake of discussion and if anyone is researching this as I was, the below is the minimum implementation I've arrived at, based on https://github.com/confluentinc/confluent-kafka-dotnet/tree/master/examples/AvroGeneric. .AddConsumer(
consumer => consumer
// ...
.AddMiddlewares(
middlewares => middlewares
.AddSchemaRegistryAvroGenericRecordDeserializer()
.AddTypedHandlers(
handlers => handlers
.AddHandler<AvroGenericRecordMessageHandler>())
)
)
/// <summary>
/// Beware, this acts as a catch-all - it handles all Avro messages, even specific ones!
/// </summary>
public sealed class AvroGenericRecordMessageHandler : IMessageHandler<GenericRecord>
{
public Task Handle(IMessageContext context, GenericRecord message)
{
Console.WriteLine(
"Partition: {0} | Offset: {1} | Message: {2} | Avro Generic",
context.ConsumerContext.Partition,
context.ConsumerContext.Offset,
message);
return Task.CompletedTask;
}
}
public sealed class SchemaRegistryGenericRecordTypeResolver : IMessageTypeResolver
{
public ValueTask<Type> OnConsumeAsync(IMessageContext context) => ValueTask.FromResult(typeof(GenericRecord));
public ValueTask OnProduceAsync(IMessageContext context) => default;
}
public static class ConsumerConfigurationBuilderExtensions
{
public static IConsumerMiddlewareConfigurationBuilder AddSchemaRegistryAvroGenericRecordDeserializer(
this IConsumerMiddlewareConfigurationBuilder middlewares)
{
return middlewares.Add(
resolver => new DeserializerConsumerMiddleware(
new ConfluentAvroDeserializer(resolver),
new SchemaRegistryGenericRecordTypeResolver()));
}
}
public static class SampleData
{
private static readonly RecordSchema s_schema = (RecordSchema)RecordSchema.Parse(
@"{
""type"": ""record"",
""name"": ""User"",
""fields"": [
{""name"": ""name"", ""type"": ""string""},
{""name"": ""favorite_number"", ""type"": ""long""},
{""name"": ""favorite_color"", ""type"": ""string""}
]
}"
);
public static GenericRecord CreateGenericUser(string name, long favouriteNumber, string favouriteColor)
{
var record = new GenericRecord(s_schema);
record.Add("name", name);
record.Add("favorite_number", favouriteNumber);
record.Add("favorite_color", favouriteColor);
return record;
}
} The main issue that wasn't very obvious is that the existing Question to the maintainers of the library, do you see value in adding this natively into the lib, or at the very least showcasing this in the samples and/or docs? If this is to be added to the lib, an API must be defined, because there's more than one way of doing it. Above I've demonstrated a new deserializer, which is simple enough, but you may wish not to introduce a different registration method. Another option is to extend the existing .AddConsumer(
consumer => consumer
// ...
.AddMiddlewares(
middlewares => middlewares
.AddSchemaRegistryAvroGenericRecordDeserializer()
.AddTypedHandlers(
handlers => handlers
.AddHandler<AvroMessageHandler>()
.AddHandler<AvroMessageHandler2>()
// everything that didn't match specific handlers will fall through to the generic handler
// assumes we modified the SchemaRegistryTypeResolver to fallback gracefully to typeof(GenericRecord)
.AddHandler<AvroGenericRecordMessageHandler>())
)
) Whilst I haven't tested the above design, I don't see a reason why it wouldn't work. |
Hi @danielmpetrov, Thank you for your suggestion, and we see value on adding this to KafkaFlow. Since this will require a new implementation on KafkaFlow, we will work on it, and then open the PR. Feel free to participate on it when we open the PR. |
Awesome, I'm looking forward to your design. |
Is your request related to a problem you have?
I was running some samples in the
Confluent.Kafka
repo, and then some samples in this repo, and since I was interested in aGenericRecord
example with Avro it made an impression on me that there are two Avro examples, and both useISpecificRecord
. In Contrast in theConfluent.Kafka
repo they have one example for specific and one for generic, which is much more valuable in my opinion. See https://github.com/confluentinc/confluent-kafka-dotnet/tree/master/examples/AvroGeneric and https://github.com/confluentinc/confluent-kafka-dotnet/tree/master/examples/AvroSpecific for comparison.Describe the solution you'd like
Add a generic avro record sample, in addition to the two specific record examples. Possibly replace one of the specific example with a generic for better value. Refer to https://github.com/confluentinc/confluent-kafka-dotnet/tree/master/examples/AvroGeneric.
Are you able to help bring it to life and contribute with a Pull Request?
Yes
Additional context
I might be able to make a PR for this, but first I must try it for myself. Hints appreciated.
The text was updated successfully, but these errors were encountered: