From 2510845edb696147ef7ee15f694af552ac4d437b Mon Sep 17 00:00:00 2001 From: Bingfeng Xia Date: Wed, 29 Jul 2020 13:56:45 -0700 Subject: [PATCH] Added a config to limit the code-gen and class loading It sets the maximum number of fast SerDes classes generated and loaded. It helps to limit the metaspace and codecache usage brought by fast-avro runtime code-gen and class loading. The limit config can be set via FastSerdeCache constructors. --- .../fastserde/FastDeserializerGenerator.java | 5 ++ .../FastDeserializerGeneratorBase.java | 7 ++ .../FastGenericDeserializerGenerator.java | 5 ++ .../FastGenericSerializerGenerator.java | 5 ++ .../avro/fastserde/FastSerdeBase.java | 35 +++++++- .../avro/fastserde/FastSerdeCache.java | 89 +++++++++++++++++-- .../fastserde/FastSerializerGenerator.java | 6 ++ .../FastSpecificDeserializerGenerator.java | 5 ++ .../FastSpecificSerializerGenerator.java | 5 ++ .../avro/fastserde/FastDatumReaderTest.java | 47 ++++++++++ .../FastGenericSerializerGeneratorTest.java | 21 +++++ 11 files changed, 221 insertions(+), 9 deletions(-) diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastDeserializerGenerator.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastDeserializerGenerator.java index 56785f221..6722713c6 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastDeserializerGenerator.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastDeserializerGenerator.java @@ -71,6 +71,11 @@ public class FastDeserializerGenerator extends FastDeserializerGeneratorBase< super(useGenericTypes, writer, reader, destination, classLoader, compileClassPath); } + FastDeserializerGenerator(boolean useGenericTypes, Schema writer, Schema reader, File destination, + ClassLoader classLoader, String compileClassPath, int loadClassLimit) { + super(useGenericTypes, writer, reader, destination, classLoader, compileClassPath, loadClassLimit); + } + public FastDeserializer generateDeserializer() { String className = getClassName(writer, reader, useGenericTypes ? "Generic" : "Specific"); JPackage classPackage = codeModel._package(generatedPackageName); diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastDeserializerGeneratorBase.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastDeserializerGeneratorBase.java index d3cd4079d..8b4794202 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastDeserializerGeneratorBase.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastDeserializerGeneratorBase.java @@ -25,6 +25,13 @@ public abstract class FastDeserializerGeneratorBase extends FastSerdeBase { this.reader = reader; } + FastDeserializerGeneratorBase(boolean useGenericTypes, Schema writer, Schema reader, File destination, ClassLoader classLoader, + String compileClassPath, int loadClassLimit) { + super("deserialization", useGenericTypes, Utf8.class, destination, classLoader, compileClassPath, false, loadClassLimit); + this.writer = writer; + this.reader = reader; + } + protected static Symbol[] reverseSymbolArray(Symbol[] symbols) { Symbol[] reversedSymbols = new Symbol[symbols.length]; diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastGenericDeserializerGenerator.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastGenericDeserializerGenerator.java index fa2661440..85bbcc89b 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastGenericDeserializerGenerator.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastGenericDeserializerGenerator.java @@ -10,4 +10,9 @@ public final class FastGenericDeserializerGenerator extends FastDeserializerG String compileClassPath) { super(true, writer, reader, destination, classLoader, compileClassPath); } + + FastGenericDeserializerGenerator(Schema writer, Schema reader, File destination, ClassLoader classLoader, + String compileClassPath, int loadClassLimit) { + super(true, writer, reader, destination, classLoader, compileClassPath, loadClassLimit); + } } diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastGenericSerializerGenerator.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastGenericSerializerGenerator.java index b99517967..032744c90 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastGenericSerializerGenerator.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastGenericSerializerGenerator.java @@ -10,4 +10,9 @@ public FastGenericSerializerGenerator(Schema schema, File destination, ClassLoad String compileClassPath) { super(true, schema, destination, classLoader, compileClassPath); } + + public FastGenericSerializerGenerator(Schema schema, File destination, ClassLoader classLoader, + String compileClassPath, int loadClassLimit) { + super(true, schema, destination, classLoader, compileClassPath, loadClassLimit); + } } diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerdeBase.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerdeBase.java index 36e33a862..74516a6ee 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerdeBase.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerdeBase.java @@ -14,6 +14,7 @@ import java.util.concurrent.ConcurrentMap; import java.util.concurrent.atomic.AtomicInteger; import java.util.function.Consumer; +import java.util.function.Supplier; import javax.tools.JavaCompiler; import javax.tools.ToolProvider; import org.apache.avro.Schema; @@ -32,6 +33,9 @@ public abstract class FastSerdeBase { protected static final String SEP = "_"; public static final String GENERATED_PACKAGE_NAME_PREFIX = "com.linkedin.avro.fastserde.generated."; + private volatile int loadClassLimit = Integer.MAX_VALUE; + private int loadClassNum = 0; + /** * A repository of how many times a given name was used. * N.B.: Does not actually need to be threadsafe, but it is made so just for defensive coding reasons. @@ -47,6 +51,12 @@ public abstract class FastSerdeBase { protected final String compileClassPath; protected JDefinedClass generatedClass; + public FastSerdeBase(String description, boolean useGenericTypes, Class defaultStringClass, File destination, ClassLoader classLoader, + String compileClassPath, boolean isForSerializer, int loadClassLimit) { + this(description, useGenericTypes, defaultStringClass, destination, classLoader, compileClassPath, isForSerializer); + this.loadClassLimit = loadClassLimit; + } + public FastSerdeBase(String description, boolean useGenericTypes, Class defaultStringClass, File destination, ClassLoader classLoader, String compileClassPath, boolean isForSerializer) { this.useGenericTypes = useGenericTypes; @@ -136,6 +146,29 @@ protected Class compileClass(final String className, Set knownUsedFullyQ throw new FastSerdeGeneratorException("Unable to compile:" + className + " from source file: " + filePath); } - return classLoader.loadClass(generatedPackageName + "." + className); + return loadClassWithLimit(() -> { + try { + return classLoader.loadClass(generatedPackageName + "." + className); + } catch (ClassNotFoundException e) { + throw new FastSerdeGeneratorException("Unable to load:" + className + " from source file: " + filePath, e); + } + }); + } + + /** + * A wrapper function that limits the total number of fast de/serializer classes loaded + * + * @param supplier The function to load fast de/serializer class + */ + protected synchronized T loadClassWithLimit(Supplier supplier) { + if (this.loadClassNum < this.loadClassLimit) { + T classObj = null; + classObj = supplier.get(); + this.loadClassNum++; + return classObj; + } else { + LOGGER.warn("Loaded fast serdes classes number {}, with limit set to {}", loadClassNum, loadClassLimit); + } + return null; } } diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerdeCache.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerdeCache.java index a0a61c201..bb7ef46cd 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerdeCache.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerdeCache.java @@ -44,6 +44,20 @@ public final class FastSerdeCache { private static volatile FastSerdeCache _INSTANCE; + /** + * Fast-avro will generate and load serializer and deserializer(SerDes) classes into metaspace during runtime. + * During serialization and deserialization, fast-avro also leverages JIT compilation to boost the SerDes speed. + * And JIT compilation code is saved in code cache. + * Too much usage of metaspace and code cache will bring GC/OOM issue. + * + * We set a hard limit of the total number of SerDes classes generated and loaded by fast-avro. + * By default, the limit is set to MAX_INT. + * Fast-avro will fall back to regular avro after the limit is hit. + * One could set the limit through {@link FastSerdeCache} constructors. + */ + private volatile int generatedFastSerDesLimit = Integer.MAX_VALUE; + private final AtomicInteger generatedSerDesNum = new AtomicInteger(0); + private final Map> fastSpecificRecordDeserializersCache = new FastAvroConcurrentHashMap<>(); private final Map> fastGenericRecordDeserializersCache = @@ -81,6 +95,20 @@ public FastSerdeCache(Executor executorService, Supplier compileClassPat this(executorService, compileClassPathSupplier.get()); } + /** + * + * @param executorService + * {@link Executor} used by serializer/deserializer compile threads + * @param compileClassPathSupplier + * custom classpath {@link Supplier} + * @param limit + * custom number {@link #generatedFastSerDesLimit} + */ + public FastSerdeCache(Executor executorService, Supplier compileClassPathSupplier, int limit) { + this(executorService, compileClassPathSupplier); + this.generatedFastSerDesLimit = limit; + } + public FastSerdeCache(String compileClassPath) { this(); this.compileClassPath = Optional.ofNullable(compileClassPath); @@ -118,10 +146,24 @@ public FastSerdeCache(Executor executorService) { this.compileClassPath = Optional.empty(); } + public FastSerdeCache(Executor executorService, int limit) { + this(executorService); + this.generatedFastSerDesLimit = limit; + } + private FastSerdeCache() { this((Executor) null); } + /** + * @param limit + * custom number {@link #generatedFastSerDesLimit} + */ + public FastSerdeCache(int limit) { + this(); + this.generatedFastSerDesLimit = limit; + } + /** * Gets default {@link FastSerdeCache} instance. Default instance classpath can be customized via * {@value #CLASSPATH} or {@value #CLASSPATH_SUPPLIER} system properties. @@ -296,6 +338,25 @@ private String getSchemaKey(Schema writerSchema, Schema readerSchema) { Utils.getSchemaFingerprint(readerSchema)); } + /** + * A wrapper function that limits the total number of fast de/serializer classes generated + * + * @param supplier The function to build and save fast de/serializer + */ + private T buildFastClassWithLimit(Supplier supplier) { + T result = null; + if (this.generatedSerDesNum.get() < this.generatedFastSerDesLimit) { + result = supplier.get(); + } else if (this.generatedSerDesNum.get() == this.generatedFastSerDesLimit) { + // We still want to print the warning when the limit is hit + LOGGER.warn("Generated fast serdes classes number hits limit {}", this.generatedFastSerDesLimit); + } else { + LOGGER.debug("Generated serdes number {}, with fast serdes limit set to {}", this.generatedSerDesNum.get(), this.generatedFastSerDesLimit); + } + generatedSerDesNum.incrementAndGet(); + return result; + } + /** * This function will generate a fast specific deserializer, and it will throw exception if anything wrong happens. * This function can be used to verify whether current {@link FastSerdeCache} could generate proper fast deserializer. @@ -307,7 +368,7 @@ private String getSchemaKey(Schema writerSchema, Schema readerSchema) { public FastDeserializer buildFastSpecificDeserializer(Schema writerSchema, Schema readerSchema) { FastSpecificDeserializerGenerator generator = new FastSpecificDeserializerGenerator<>(writerSchema, readerSchema, classesDir, classLoader, - compileClassPath.orElseGet(() -> null)); + compileClassPath.orElseGet(() -> null), generatedFastSerDesLimit); LOGGER.info("Generated class dir: {}, and generation of specific FastDeserializer is done for writer schema: " + "[\n{}\n] and reader schema: [\n{}\n]", classesDir, writerSchema.toString(true), readerSchema.toString(true)); return generator.generateDeserializer(); @@ -322,7 +383,10 @@ public FastDeserializer buildFastSpecificDeserializer(Schema writerSchema, Sc */ private FastDeserializer buildSpecificDeserializer(Schema writerSchema, Schema readerSchema) { try { - return buildFastSpecificDeserializer(writerSchema, readerSchema); + FastDeserializer fastSpecificDeserializer = buildFastClassWithLimit(() -> buildFastSpecificDeserializer(writerSchema, readerSchema)); + if (fastSpecificDeserializer != null) { + return fastSpecificDeserializer; + } } catch (FastDeserializerGeneratorException e) { LOGGER.warn("Deserializer generation exception when generating specific FastDeserializer for writer schema: " + "[\n{}\n] and reader schema: [\n{}\n]", writerSchema.toString(true), readerSchema.toString(true), e); @@ -351,7 +415,7 @@ public Object deserialize(Object reuse, Decoder d) throws IOException { public FastDeserializer buildFastGenericDeserializer(Schema writerSchema, Schema readerSchema) { FastGenericDeserializerGenerator generator = new FastGenericDeserializerGenerator<>(writerSchema, readerSchema, classesDir, classLoader, - compileClassPath.orElseGet(() -> null)); + compileClassPath.orElseGet(() -> null), generatedFastSerDesLimit); LOGGER.info("Generated classes dir: {} and generation of generic FastDeserializer is done for writer schema: " + "[\n{}\n] and reader schema:[\n{}\n]", classesDir, writerSchema.toString(true), readerSchema.toString(true)); return generator.generateDeserializer(); @@ -367,7 +431,10 @@ public FastDeserializer buildFastGenericDeserializer(Schema writerSchema, Sch */ private FastDeserializer buildGenericDeserializer(Schema writerSchema, Schema readerSchema) { try { - return buildFastGenericDeserializer(writerSchema, readerSchema); + FastDeserializer fastGenericDeserializer = buildFastClassWithLimit(() -> buildFastGenericDeserializer(writerSchema, readerSchema)); + if (fastGenericDeserializer != null) { + return fastGenericDeserializer; + } } catch (FastDeserializerGeneratorException e) { LOGGER.warn("Deserializer generation exception when generating generic FastDeserializer for writer schema: [\n" + writerSchema.toString(true) + "\n] and reader schema:[\n" + readerSchema.toString(true) + "\n]", e); @@ -392,7 +459,7 @@ public FastSerializer buildFastSpecificSerializer(Schema schema) { Utils.getAvroVersionsSupportedForSerializer()); } FastSpecificSerializerGenerator generator = - new FastSpecificSerializerGenerator<>(schema, classesDir, classLoader, compileClassPath.orElseGet(() -> null)); + new FastSpecificSerializerGenerator<>(schema, classesDir, classLoader, compileClassPath.orElseGet(() -> null), generatedFastSerDesLimit); LOGGER.info("Generated classes dir: {} and generation of specific FastSerializer is done for schema: [\n{}\n]", classesDir, schema.toString(true)); return generator.generateSerializer(); @@ -402,7 +469,10 @@ private FastSerializer buildSpecificSerializer(Schema schema) { if (Utils.isSupportedAvroVersionsForSerializer()) { // Only build fast specific serializer for supported Avro versions. try { - return buildFastSpecificSerializer(schema); + FastSerializer fastSpecificSerializer = buildFastClassWithLimit(() -> buildFastSpecificSerializer(schema)); + if (fastSpecificSerializer != null) { + return fastSpecificSerializer; + } } catch (FastDeserializerGeneratorException e) { LOGGER.warn("Serializer generation exception when generating specific FastSerializer for schema: [\n{}\n]", schema.toString(true), e); @@ -428,7 +498,7 @@ public FastSerializer buildFastGenericSerializer(Schema schema) { + Utils.getAvroVersionsSupportedForSerializer()); } FastGenericSerializerGenerator generator = - new FastGenericSerializerGenerator<>(schema, classesDir, classLoader, compileClassPath.orElseGet(() -> null)); + new FastGenericSerializerGenerator<>(schema, classesDir, classLoader, compileClassPath.orElseGet(() -> null), generatedFastSerDesLimit); LOGGER.info("Generated classes dir: {} and generation of generic FastSerializer is done for schema: [\n{}\n]", classesDir, schema.toString(true)); return generator.generateSerializer(); @@ -438,7 +508,10 @@ private FastSerializer buildGenericSerializer(Schema schema) { if (Utils.isSupportedAvroVersionsForSerializer()) { // Only build fast generic serializer for supported Avro versions. try { - return buildFastGenericSerializer(schema); + FastSerializer fastGenericSerializer = buildFastClassWithLimit(() -> buildFastGenericSerializer(schema)); + if (fastGenericSerializer != null) { + return fastGenericSerializer; + } } catch (FastDeserializerGeneratorException e) { LOGGER.warn("Serializer generation exception when generating generic FastSerializer for schema: [\n{}\n]", schema.toString(true), e); diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerializerGenerator.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerializerGenerator.java index 5dcfecdaf..0a705ac8d 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerializerGenerator.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSerializerGenerator.java @@ -42,6 +42,12 @@ public FastSerializerGenerator(boolean useGenericTypes, Schema schema, File dest this.schema = schema; } + public FastSerializerGenerator(boolean useGenericTypes, Schema schema, File destination, ClassLoader classLoader, + String compileClassPath, int loadClassLimit) { + super("serialization", useGenericTypes, CharSequence.class, destination, classLoader, compileClassPath, true, loadClassLimit); + this.schema = schema; + } + public static String getClassName(Schema schema, String description) { Long schemaId = Math.abs(Utils.getSchemaFingerprint(schema)); String typeName = SchemaAssistant.getTypeName(schema); diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSpecificDeserializerGenerator.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSpecificDeserializerGenerator.java index 13d9bf0df..a574b211a 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSpecificDeserializerGenerator.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSpecificDeserializerGenerator.java @@ -10,4 +10,9 @@ public final class FastSpecificDeserializerGenerator extends FastDeserializer String compileClassPath) { super(false, writer, reader, destination, classLoader, compileClassPath); } + + FastSpecificDeserializerGenerator(Schema writer, Schema reader, File destination, ClassLoader classLoader, + String compileClassPath, int loadClassLimit) { + super(false, writer, reader, destination, classLoader, compileClassPath, loadClassLimit); + } } diff --git a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSpecificSerializerGenerator.java b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSpecificSerializerGenerator.java index 09ff98233..529949be2 100644 --- a/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSpecificSerializerGenerator.java +++ b/avro-fastserde/src/main/java/com/linkedin/avro/fastserde/FastSpecificSerializerGenerator.java @@ -10,4 +10,9 @@ public FastSpecificSerializerGenerator(Schema schema, File destination, ClassLoa String compileClassPath) { super(false, schema, destination, classLoader, compileClassPath); } + + public FastSpecificSerializerGenerator(Schema schema, File destination, ClassLoader classLoader, + String compileClassPath, int loadClassLimit) { + super(false, schema, destination, classLoader, compileClassPath, loadClassLimit); + } } diff --git a/avro-fastserde/src/test/java/com/linkedin/avro/fastserde/FastDatumReaderTest.java b/avro-fastserde/src/test/java/com/linkedin/avro/fastserde/FastDatumReaderTest.java index 7ccaa34fd..f26df96cf 100644 --- a/avro-fastserde/src/test/java/com/linkedin/avro/fastserde/FastDatumReaderTest.java +++ b/avro-fastserde/src/test/java/com/linkedin/avro/fastserde/FastDatumReaderTest.java @@ -97,4 +97,51 @@ public void shouldCreateGenericDatumReader() throws IOException, InterruptedExce Assert.assertEquals(new Utf8("test"), fastGenericDatumReader.read(null, FastSerdeTestsSupport.genericDataAsDecoder(record)).get("test")); } + + @Test(groups = {"deserializationTest"}) + @SuppressWarnings("unchecked") + public void shouldNotCreateFastDeserializerDueToLimit() throws IOException, InterruptedException { + // Set generatedFastSerDesLimit to 1 + // Try to generate fast deserializer for the first schema + Schema recordSchema1 = createRecord("TestSchema1", createPrimitiveUnionFieldSchema("test", Schema.Type.STRING)); + GenericRecord record1 = new GenericData.Record(recordSchema1); + record1.put("test", "test"); + + FastSerdeCache cacheLimit1 = new FastSerdeCache(Runnable::run, 1); + FastGenericDatumReader fastGenericDatumReader = new FastGenericDatumReader<>(recordSchema1, cacheLimit1); + + // when + fastGenericDatumReader.read(null, FastSerdeTestsSupport.genericDataAsDecoder(record1)); + + // then + FastDeserializer fastGenericDeserializer = + (FastDeserializer) cacheLimit1.getFastGenericDeserializer(recordSchema1, recordSchema1); + + fastGenericDeserializer = + (FastDeserializer) cacheLimit1.getFastGenericDeserializer(recordSchema1, recordSchema1); + + Assert.assertNotNull(fastGenericDeserializer); + Assert.assertNotEquals(1, fastGenericDeserializer.getClass().getDeclaredMethods().length); + Assert.assertEquals(new Utf8("test"), + fastGenericDatumReader.read(null, FastSerdeTestsSupport.genericDataAsDecoder(record1)).get("test")); + + // Try to generate fast deserializer for the second schema + // Verify only return FastDeserializerWithAvroGenericImpl + Schema recordSchema2 = createRecord("TestSchema2", createPrimitiveUnionFieldSchema("test", Schema.Type.STRING)); + GenericRecord record2 = new GenericData.Record(recordSchema2); + record2.put("test", "test"); + + // when + fastGenericDatumReader.read(null, FastSerdeTestsSupport.genericDataAsDecoder(record2)); + + // then + FastDeserializer fastGenericDeserializer2 = + (FastDeserializer) cacheLimit1.getFastGenericDeserializer(recordSchema2, recordSchema2); + + fastGenericDeserializer2 = + (FastDeserializer) cacheLimit1.getFastGenericDeserializer(recordSchema2, recordSchema2); + + Assert.assertNotNull(fastGenericDeserializer2); + Assert.assertEquals(1, fastGenericDeserializer2.getClass().getDeclaredMethods().length); + } } diff --git a/avro-fastserde/src/test/java/com/linkedin/avro/fastserde/FastGenericSerializerGeneratorTest.java b/avro-fastserde/src/test/java/com/linkedin/avro/fastserde/FastGenericSerializerGeneratorTest.java index 611820d02..bb9e3dd62 100644 --- a/avro-fastserde/src/test/java/com/linkedin/avro/fastserde/FastGenericSerializerGeneratorTest.java +++ b/avro-fastserde/src/test/java/com/linkedin/avro/fastserde/FastGenericSerializerGeneratorTest.java @@ -10,6 +10,7 @@ import com.linkedin.avroutil1.compatibility.AvroCompatibilityHelper; import java.io.ByteArrayOutputStream; import java.io.File; +import java.io.IOException; import java.net.URL; import java.net.URLClassLoader; import java.nio.ByteBuffer; @@ -18,6 +19,8 @@ import java.util.HashMap; import java.util.List; import java.util.Map; +import java.nio.file.Files; +import java.nio.file.Path; import java.util.concurrent.atomic.AtomicBoolean; import org.apache.avro.Schema; import org.apache.avro.generic.GenericContainer; @@ -528,6 +531,24 @@ public long getPrimitive(int index) { Assert.assertTrue(primitiveApiCalled.get()); } + @Test(expectedExceptions = FastSerdeGeneratorException.class) + public void shouldNotGenerateFastSerializer() throws IOException { + // given + Schema recordSchema = createRecord("TestSchema"); + GenericRecord record = new GenericData.Record(recordSchema); + + // create a specific classloader for this test to avoid useless generated codes + Path tempPath = Files.createTempDirectory("generated"); + File tempDir = tempPath.toFile(); + ClassLoader tempClassLoader = URLClassLoader.newInstance(new URL[]{tempDir.toURI().toURL()}, + FastGenericSerializerGeneratorTest.class.getClassLoader()); + + // when + FastGenericSerializerGenerator fastGenericSerializerGenerator = + new FastGenericSerializerGenerator<>(recordSchema, tempDir, tempClassLoader, null, 0); + fastGenericSerializerGenerator.generateSerializer(); + } + private void shouldWriteArrayOfPrimitives(Schema.Type elementType, List data) { // given Schema elementSchema = Schema.create(elementType);