Skip to content
This repository was archived by the owner on Aug 14, 2023. It is now read-only.
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
18 changes: 18 additions & 0 deletions client/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 3 additions & 3 deletions src/main/java/org/akhq/models/Record.java
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,12 @@
import com.fasterxml.jackson.annotation.JsonIgnore;
import io.confluent.kafka.serializers.KafkaAvroDeserializer;
import lombok.*;
import org.akhq.utils.AvroToJsonSerializer;
import org.apache.avro.generic.GenericRecord;
import org.apache.kafka.clients.consumer.ConsumerRecord;
import org.apache.kafka.clients.producer.RecordMetadata;
import org.apache.kafka.common.header.Header;
import org.apache.kafka.common.record.TimestampType;
import org.joda.time.DateTime;

import java.nio.ByteBuffer;
import java.time.Instant;
Expand Down Expand Up @@ -107,8 +107,8 @@ private String convertToString(byte[] payload, Integer keySchemaId) {
return null;
} else if (keySchemaId != null) {
try {
GenericRecord deserialize = (GenericRecord) kafkaAvroDeserializer.deserialize(topic, payload);
return deserialize.toString();
GenericRecord record = (GenericRecord) kafkaAvroDeserializer.deserialize(topic, payload);
return AvroToJsonSerializer.toJson(record);
} catch (Exception exception) {
return new String(payload);
}
Expand Down
8 changes: 4 additions & 4 deletions src/main/java/org/akhq/models/Schema.java
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,8 @@
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.google.common.annotations.VisibleForTesting;
import lombok.*;
import org.akhq.utils.AvroDeserializer;
import org.akhq.utils.AvroSerializer;
import org.akhq.utils.AvroSchemaDeserializer;
import org.akhq.utils.AvroSchemaSerializer;
import org.apache.avro.AvroTypeException;
import org.apache.avro.Schema.Parser;

Expand All @@ -28,8 +28,8 @@ public class Schema {
private Integer version;
private Config.CompatibilityLevelConfig compatibilityLevel;

@JsonSerialize(using = AvroSerializer.class)
@JsonDeserialize(using = AvroDeserializer.class)
@JsonSerialize(using = AvroSchemaSerializer.class)
@JsonDeserialize(using = AvroSchemaDeserializer.class)
private org.apache.avro.Schema schema;

private String exception;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@

import java.io.IOException;

public class AvroDeserializer extends JsonDeserializer<Schema> {
public class AvroSchemaDeserializer extends JsonDeserializer<Schema> {
@Override
public Schema deserialize(
JsonParser p,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@

import java.io.IOException;

public class AvroSerializer extends JsonSerializer<Schema> {
public class AvroSchemaSerializer extends JsonSerializer<Schema> {
@Override
public void serialize(
Schema value,
Expand Down
63 changes: 63 additions & 0 deletions src/main/java/org/akhq/utils/AvroToJsonSerializer.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
package org.akhq.utils;

import org.apache.avro.*;
import org.apache.avro.generic.GenericData;
import org.apache.avro.generic.GenericRecord;
import org.apache.avro.io.Encoder;
import org.apache.avro.io.EncoderFactory;
import org.apache.avro.io.JsonEncoder;
import org.apache.avro.specific.SpecificDatumWriter;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.math.BigDecimal;
import java.nio.ByteBuffer;

public class AvroToJsonSerializer {

public static String toJson(GenericRecord record) throws IOException {
try (ByteArrayOutputStream outputStream = new ByteArrayOutputStream()) {
JsonEncoder jsonEncoder = EncoderFactory.get().jsonEncoder(record.getSchema(), outputStream);
new BigDecimalFriendlySpecificDatumWriter<GenericRecord>(record.getSchema()).write(record, jsonEncoder);
jsonEncoder.flush();
return new String(outputStream.toByteArray());
}
}
}

class BigDecimalFriendlySpecificDatumWriter<T> extends SpecificDatumWriter<T> {

private static final Conversion<BigDecimal> DECIMAL_CONVERSION = new Conversions.DecimalConversion();

public BigDecimalFriendlySpecificDatumWriter(Schema schema) {
super(schema);
}

@Override
protected void writeField(Object datum, Schema.Field f, Encoder out, Object state) throws IOException {
if (datum instanceof GenericData.Record) {
Schema fieldSchema = f.schema();
LogicalType logicalType = fieldSchema.getLogicalType();
Object value = getData().getField(datum, f.name(), f.pos());
if (logicalType instanceof LogicalTypes.Decimal) {
value = convert(DECIMAL_CONVERSION, fieldSchema, logicalType, value);
}
writeWithoutConversion(fieldSchema, value, out);
} else {
super.writeField(datum, f, out, state);
}
}

private Object convert(Conversion<?> conversion, Schema fieldSchema, LogicalType logicalType, Object value) {
if (conversion instanceof Conversions.DecimalConversion && value instanceof ByteBuffer) {
// convert decimal value to a string
byte[] byteValue = new byte[((ByteBuffer) value).remaining()];
((ByteBuffer) value).get(byteValue);
BigDecimal number = (BigDecimal) conversion.fromBytes(ByteBuffer.wrap(byteValue), fieldSchema, logicalType);
return (ByteBuffer.wrap(number.toPlainString().getBytes()));
} else {
return convert(fieldSchema, logicalType, conversion, value);
}
}
}

10 changes: 10 additions & 0 deletions src/test/avro/Dog.avsc
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
{
"name": "Dog",
"namespace": "org.akhq",
"type": "record",
"fields" : [
{"name": "id", "type": "int"},
{"name": "name", "type": "string"},
{"name": "weight", "type" : { "type" : "bytes", "logicalType" : "decimal", "precision" : 4, "scale" : 2}}
]
}
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@
import static org.mockito.Mockito.when;

@ExtendWith(MockitoExtension.class)
class AvroSerializerTest {
class AvroSchemaSerializerTest {

private final org.apache.avro.Schema SCHEMA = SchemaBuilder
.record("schema1").namespace("org.akhq")
Expand Down
54 changes: 54 additions & 0 deletions src/test/java/org/akhq/modules/AvroToJsonSerializerTest.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
package org.akhq.modules;

import org.akhq.Breed;
import org.akhq.Cat;
import org.akhq.Dog;
import org.akhq.utils.AvroToJsonSerializer;
import org.apache.avro.Conversion;
import org.apache.avro.Conversions;
import org.apache.avro.Schema;
import org.apache.avro.generic.GenericRecord;
import org.apache.avro.generic.GenericRecordBuilder;
import org.junit.jupiter.api.Test;

import java.io.IOException;
import java.math.BigDecimal;

import static org.junit.jupiter.api.Assertions.assertEquals;

public class AvroToJsonSerializerTest {

private static final Conversion<BigDecimal> DECIMAL_CONVERSION = new Conversions.DecimalConversion();

@Test
public void serializeAvroToJsonWithDecimal() throws IOException {
String expectedString = "{\"id\":10,\"name\":\"Tiger\",\"weight\":\"10.40\"}";
GenericRecord dogExample = aDogExample(10, "Tiger", 10.40);
assertEquals(AvroToJsonSerializer.toJson(dogExample), expectedString);
}

@Test
public void serializeAvroToJsonWithoutDecimal() throws IOException {
String expectedString = "{\"id\":10,\"name\":\"Tom\",\"breed\":\"SPHYNX\"}";
GenericRecord catExample = aCatExample(10, "Tom", Breed.SPHYNX);
assertEquals(AvroToJsonSerializer.toJson(catExample), expectedString);
}

private GenericRecord aCatExample(int id, String name, Breed breed) {
return new GenericRecordBuilder(Cat.SCHEMA$)
.set("id", id)
.set("name", name)
.set("breed", breed)
.build();
}

private GenericRecord aDogExample(int id, String name, double weight) {
Schema.Field weightField = Dog.SCHEMA$.getField("weight");
return new GenericRecordBuilder(Dog.SCHEMA$)
.set("id", id)
.set("name", name)
.set("weight", DECIMAL_CONVERSION.toBytes(BigDecimal.valueOf(weight).setScale(2), weightField.schema(), weightField.schema().getLogicalType()))
.build();
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -122,7 +122,7 @@ public void consumeAvro() throws ExecutionException, InterruptedException {
.findFirst();

avroRecord.orElseThrow(() -> new NoSuchElementException("Unable to find key 1"));
avroRecord.ifPresent(record -> assertEquals("{\"id\": 1, \"name\": \"WaWa\", \"breed\": \"ABYSSINIAN\"}", record.getValue()));
avroRecord.ifPresent(record -> assertEquals("{\"id\":1,\"name\":\"WaWa\",\"breed\":\"ABYSSINIAN\"}", record.getValue()));
}

private List<Record> consumeAllRecord(RecordRepository.Options options) throws ExecutionException, InterruptedException {
Expand Down