Je cherche un moyen de convertir un POJO en objet avro de manière générique. L'implémentation doit être robuste à tout changement de la classe POJO. Je l'ai atteint mais en remplissant explicitement l'enregistrement avro (voir l'exemple ci-dessous).
Existe-t-il un moyen de se débarrasser des noms de champ codés en dur et de simplement remplir l'enregistrement avro de l'objet? La réflexion est-elle le seul moyen ou avro fournit-elle cette fonctionnalité prête à l'emploi?
import Java.util.Date;
import Java.util.HashMap;
import Java.util.Map;
import org.Apache.avro.Schema;
import org.Apache.avro.generic.GenericData.Record;
import org.Apache.avro.reflect.ReflectData;
public class PojoToAvroExample {
static class PojoParent {
public final Map<String, String> aMap = new HashMap<String, String>();
public final Map<String, Integer> anotherMap = new HashMap<String, Integer>();
}
static class Pojo extends PojoParent {
public String uid;
public Date eventTime;
}
static Pojo createPojo() {
Pojo foo = new Pojo();
foo.uid = "123";
foo.eventTime = new Date();
foo.aMap.put("key", "val");
foo.anotherMap.put("key", 42);
return foo;
}
public static void main(String[] args) {
// extract the avro schema corresponding to Pojo class
Schema schema = ReflectData.get().getSchema(Pojo.class);
System.out.println("extracted avro schema: " + schema);
// create avro record corresponding to schema
Record avroRecord = new Record(schema);
System.out.println("corresponding empty avro record: " + avroRecord);
Pojo foo = createPojo();
// TODO: to be replaced by generic variant:
// something like avroRecord.importValuesFrom(foo);
avroRecord.put("uid", foo.uid);
avroRecord.put("eventTime", foo.eventTime);
avroRecord.put("aMap", foo.aMap);
avroRecord.put("anotherMap", foo.anotherMap);
System.out.println("expected avro record: " + avroRecord);
}
}
Utilisez-vous Spring?
Je construis un mappeur pour cela en utilisant une fonctionnalité Spring. Mais il est également possible de construire un tel mappeur via des utilitaires de réflexion bruts:
import org.Apache.avro.Schema;
import org.Apache.avro.generic.GenericData;
import org.Apache.avro.reflect.ReflectData;
import org.springframework.beans.PropertyAccessorFactory;
import org.springframework.util.Assert;
public class GenericRecordMapper {
public static GenericData.Record mapObjectToRecord(Object object) {
Assert.notNull(object, "object must not be null");
final Schema schema = ReflectData.get().getSchema(object.getClass());
final GenericData.Record record = new GenericData.Record(schema);
schema.getFields().forEach(r -> record.put(r.name(), PropertyAccessorFactory.forDirectFieldAccess(object).getPropertyValue(r.name())));
return record;
}
public static <T> T mapRecordToObject(GenericData.Record record, T object) {
Assert.notNull(record, "record must not be null");
Assert.notNull(object, "object must not be null");
final Schema schema = ReflectData.get().getSchema(object.getClass());
Assert.isTrue(schema.getFields().equals(record.getSchema().getFields()), "Schema fields didn't match");
record.getSchema().getFields().forEach(d -> PropertyAccessorFactory.forDirectFieldAccess(object).setPropertyValue(d.name(), record.get(d.name()) == null ? record.get(d.name()) : record.get(d.name()).toString()));
return object;
}
}
Avec ce mappeur, vous pouvez générer un GenericData.Record qui peut être facilement sérialisé en avro. Lorsque vous désérialisez un Avro ByteArray, vous pouvez l'utiliser pour reconstruire un POJO à partir d'un enregistrement désérialisé:
Sérialiser
byte[] serialized = avroSerializer.serialize("topic", GenericRecordMapper.mapObjectToRecord(yourPojo));
Désérialiser
GenericData.Record deserialized = (GenericData.Record) avroDeserializer.deserialize("topic", serialized);
YourPojo yourPojo = GenericRecordMapper.mapRecordToObject(deserialized, new YourPojo());
Voici une façon générique de convertir
public static <V> byte[] toBytesGeneric(final V v, final Class<V> cls) {
final ByteArrayOutputStream bout = new ByteArrayOutputStream();
final Schema schema = ReflectData.get().getSchema(cls);
final DatumWriter<V> writer = new ReflectDatumWriter<V>(schema);
final BinaryEncoder binEncoder = EncoderFactory.get().binaryEncoder(bout, null);
try {
writer.write(v, binEncoder);
binEncoder.flush();
} catch (final Exception e) {
throw new RuntimeException(e);
}
return bout.toByteArray();
}
public static void main(String[] args) {
PojoClass pojoObject = new PojoClass();
toBytesGeneric(pojoObject, PojoClass.class);
}
Avec jackson/avro , il est très facile de convertir pojo en octet [], similaire à jackson/json:
byte[] avroData = avroMapper.writer(schema).writeValueAsBytes(pojo);
p.s.
jackson gère non seulement JSON, mais aussi XML/Avro/Protobuf/YAML, etc., avec des classes et des API très similaires.
En plus de mon commentaire à @TranceMaster, la version modifiée ci-dessous fonctionne pour moi avec les types primitifs et Java sets:
import org.Apache.avro.Schema;
import org.Apache.avro.generic.GenericData;
import org.Apache.avro.reflect.ReflectData;
import org.springframework.beans.PropertyAccessorFactory;
import org.springframework.util.Assert;
public class GenericRecordMapper {
public static GenericData.Record mapObjectToRecord(Object object) {
Assert.notNull(object, "object must not be null");
final Schema schema = ReflectData.get().getSchema(object.getClass());
System.out.println(schema);
final GenericData.Record record = new GenericData.Record(schema);
schema.getFields().forEach(r -> record.put(r.name(), PropertyAccessorFactory.forDirectFieldAccess(object).getPropertyValue(r.name())));
return record;
}
public static <T> T mapRecordToObject(GenericData.Record record, T object) {
Assert.notNull(record, "record must not be null");
Assert.notNull(object, "object must not be null");
final Schema schema = ReflectData.get().getSchema(object.getClass());
Assert.isTrue(schema.getFields().equals(record.getSchema().getFields()), "Schema fields didn't match");
record
.getSchema()
.getFields()
.forEach(field ->
PropertyAccessorFactory
.forDirectFieldAccess(object)
.setPropertyValue(field.name(), record.get(field.name()))
);
return object;
}
}